Índice

El presente notebook contiene información relacionada al entendimiento de los datos y resultados obtenidos de un modelo de predicción para el problema de rehospitalización.

  1. Generalidades
  2. Entendimiento de los datos
  3. Análisis de Registros Pérdidos
  4. Análisis Exploratorio
    1. Análisis univariado - variables continuas
    2. Análisis univariado - variables cardinales
  5. Análisis de clasificación binaria usando WOE y el IV
  6. Modelo
    1. SMOTE - Balanceo de categoria minoritaria
    2. Ajuste del modelo y Estimación de parámetros
    3. Prediciendo y evaluando el desempeño del modelo
  7. Resultados y Conclusiones
  8. Recomendaciones y Estrategias
Haciendo click sobre cada una de las secciones puede ir directamente a cada una de ellas. Al finalizar cada sección encontrará un link para volver al índice.

Generalidades

El objetivo es desarrollar e implementar un modelo de predicción de rehospitalizaciones para apoyar los programas de evitabilidad post-hospitalaria. El análisis se realizará con información que describe las características sociodemográficas del individuo y con algunos datos recolectados por el personal hospitalario; para un periodo de tiempo de dos años y medio, que va desde 2016 hasta 2018.

Volver al Índice

Entendimiento de los datos

El archivo contiene registros que corresponden a eventos de rehospitalizaciones y se encuentra detallado a nivel de cada evento hospitalario. En total son 34898 registros, 18 variables, descartando de manera inicial, aquellos atributos que se derivan después del segundo diagnóstico; los datos se describen a continuación:

Generamos la estadística descriptiva; en ella se puede visualizar que será necesario realizar más adelante algunas conversiones en los tipos de datos que vienen por defecto (por ejemplo el estrato aparece como una variable numérica). Pero antes de continuar con la codificación, procederemos a observar como se encuentran nuestras variables.


Skim summary statistics
 n obs: 34746 
 n variables: 13 

-- Variable type:factor --------------------------------------------------------
  variable missing complete     n n_unique                                  top_counts ordered
 dias_hosp       0    34746 34746        6 [1]: 10775, [2]: 7755, [3]: 4940, [5,: 4311   FALSE
      edad       0    34746 34746        7  31-: 7462, 41-: 6186, 51-: 5924, 18-: 4986   FALSE
 est_civil       0    34746 34746        6       C: 14284, Sin: 9055, S: 7340, U: 1568   FALSE
   estrato       0    34746 34746        7       Sin: 11382, 6: 7203, 5: 5500, 4: 4683   FALSE
    genero       0    34746 34746        2                   F: 20427, M: 14319, NA: 0   FALSE
    marcas       0    34746 34746        3       [0,: 32890, (2,: 1799, (4,: 57, NA: 0   FALSE
 pago_hosp       0    34746 34746       11  [38: 3507, [28: 3506, [76: 3502, [21: 3496   FALSE
    quirur       0    34746 34746        2                 No: 17833, Si: 16913, NA: 0   FALSE
      ramo       0    34746 34746        2                 26: 22430, 79: 12316, NA: 0   FALSE

-- Variable type:integer -------------------------------------------------------
 variable missing complete     n    mean       sd p0     p25     p50      p75  p100     hist
       id       0    34746 34746 17373.5 10030.45  1 8687.25 17373.5 26059.75 34746 ▇▇▇▇▇▇▇▇

-- Variable type:numeric -------------------------------------------------------
   variable missing complete     n  mean   sd p0 p25 p50 p75 p100     hist
   dias_uce       0    34746 34746 3.21  4.15  0   1   2   3   37 ▇▁▁▁▁▁▁▁
   dias_uci       0    34746 34746 4.75  5.93  0   2   3   5   34 ▇▂▁▁▁▁▁▁
 rehosp_oms       0    34746 34746 0.023 0.15  0   0   0   0    1 ▇▁▁▁▁▁▁▁

Generalidades

En la gráfica siguiente podemos observar que hay en total 4 variables que contienen registros vacios: estrato, estado civil, ingreso y proveedor.

A nivel individual el porcentaje de valores perdidos para todos los casos es superior al 25%. De forma combinada hay más del 30% de campos vacíos; por ende no podemos decir que la probabilidad de que falte un valor depende solo del valor observado, y usar un método para imputarlo (la forma no es aleatoria).

Dado lo anterior, se hace necesario construir una tercera categoría, por lo menos para las variables que poseen menos campos vacíos (estrato y estado civil).

Para estimar si existe una asociación entre las variables que pueda derivarse en colinealidad, se procede primero a verificar que las variables no poseen una distribución normal, una vez realizado esto, se elige el test de Spearman para hallar la correlación lineal por atributo.

Los resultados confirman que ninguna de las variables pesenta una distribución normal y las correlaciones relacionadas a continuación, verifican posibles asociaciones entre las variables de los días en que el paciente estuvo internado en la Unidad de Cuidados Intensivos, en la Unidad de Cuidados Especiales y los días que el paciente estuvo hospitalizado. Por conocimiento de facto, la relación entre la variable “dias_uci” y “dias_uce” es entendible, ya que cuando un paciente que ha pasado por la Unidad de Cuidados Intensivos pasó su momento de crisis y su estado de salud es más estable, suele ser remitido a la Unidad de Cuidados Especiales.

Sin embargo, las correlaciones obtenidas no cumplen un umbral suficiente para considerarlas importantes, por ende se procede a conservarlas y evaluar más adelante si es preciso eliminarlas definitivamente. Por otro lado, la variable categoría y diagnóstico están altamente correlacionadas con la variable endógena, por lo que es necesario eliminarlas del análisis, para no incurrir en posibles sobreajustes en la etapa de modelado.

Como se había mencionado anteriormente, teniendo en cuenta el análisis de datos perdidos o nulos, se decide descartar la variable ingreso ya que contiene mas de un 30% en datos perdidos.

Classes ‘tbl_df’, ‘tbl’ and 'data.frame':   34746 obs. of  12 variables:
 $ edad      : Factor w/ 7 levels "18-30","31-40",..: 3 4 7 1 5 3 2 7 2 7 ...
 $ estrato   : Factor w/ 7 levels "1","2","3","4",..: 7 4 5 3 7 6 7 7 4 4 ...
 $ est_civil : Factor w/ 6 levels "C","D","S","Sin Informacion",..: 1 1 4 4 1 1 1 1 1 4 ...
 $ genero    : Factor w/ 2 levels "F","M": 1 2 1 1 2 1 2 2 1 1 ...
 $ marcas    : Factor w/ 3 levels "[0,2]","(2,4]",..: 1 1 1 1 1 1 1 1 1 1 ...
 $ ramo      : Factor w/ 2 levels "26","79": 1 1 1 2 1 1 1 1 1 2 ...
 $ quirur    : Factor w/ 2 levels "No","Si": 2 2 2 1 1 2 2 2 1 2 ...
 $ dias_hosp : num  4 3 52 2 6 2 5 14 4 1 ...
 $ dias_uci  : num  0 0 0 0 0 0 0 0 0 0 ...
 $ dias_uce  : num  0 0 0 0 0 0 0 0 0 0 ...
 $ pago_hosp : num  20604175 98000 2090823 1459979 246683 ...
 $ rehosp_oms: num  0 0 0 0 0 0 0 0 0 0 ...
Volver al Índice

Análisis Exploratorio

Análisis univariado - variables continuas

Es evidente la existencia también, de valores atípicos muy marcados tanto en el numéro de días de hospitalización, como en los números de días que el paciente estuvo en la Unidad de Cuidado Intensivo y Especial, en dónde los valores atípicos más grandes suceden en los eventos que no desencadenaron en rehospitalización.

Con el análisis anterior no sólo se logra identificar la presencia de valores atípicos, sino que también es posible evidenciar que los datos se encuentran altamente desbalanceados. En el caso de los outliers se truncará en los casos en que sea necesario, imputando los valores que superen cierto límite en el percentil, tanto mayor como menor.

Se puede observar que en la variable días UCI, correspondiente al primer diagnóstico, no parece haber una diferencia significativa en la distribución al discriminar por la variable objetivo binaria, es decir, entre los casos de rehospitalización (1) y casos de no rehospitalización (0). Adicionalmente, la distibución en ambas variables no es simétrica.

A pesar de que los datos se encuentran bastante dispersos, se logra identificar diferencias en la variable del pago -con valores más altos en los caso en que no terminó de buevo hospitalizado, y en el casó del número de días hospitalizado los rangos son mucho más pequeños cuando hay una rehospitalización.


Análisis variables categóricas

Observando las variables categóricas la diferencia entre la probabilidad de que el evento ocurra (haya rehospitalización) o no, se puede evidenciar sólo en algunas clases por categoría, pero en general, las proporciones suelen ser bastantes similares, por lo que no es posible elaborar a priori una hipótesis que estipule diferencias significativas en las distribuciones, por lo menos para ninguna de las dos variables relacionadas en el gráfico a continuación.

Por otro lado, el atributo que indica el hecho de que se hayan realizado procedimientos quirúrgicos durante la primera hospitalización muestran cierta diferencia en la distribuión por grupo; es más probable que la persona deba ser rehospitalizada de nuevo.

Con el objetivo de enriquecer el análisis exploratorio, se calcularán dos medidas muy comúnes de la teoría de la información, éstas permiten inferir algo del poder predictivo que pueden tener las variables independientes, antes de hacer parte de un modelo.

Volver al Índice

Análisis de clasificación binaria usando WOE y el IV

El peso de la evidencia (WOE) y el valor de la información (IV) ayudan, entre otras cosas, a determinar la contribución independiente de cada variable al resultado, y detectar relaciones lineales y no lineales. El WOE mide la relación entre la variable predictiva y el objeto binario, mientras que el IV mide la fuerza predictiva de esa relación.

La tabla a continuación contiene los valores del “valor de la información” con y sin el ajuste derivado de la validación cruzada. Cuando se realiza el ajuste con el objetivo de que los resultados sean más estables, tanto pago del diagnóstico, el hecho de que el paciente halla pasado por la Unidad de cuidados, y si fueron realizados procedimientos quirúrgicos serán las únicas variables con suficiente capacidad de predicción a nivel individual y univariable (Iv > 0.05). Cuando se relaja el supuesto, IV sin restar el penalty, se incluirían los días en que estuvo hospitalizado.

Variable IV PENALTY AdjIV
10 dias_uce 3.2493169 0.5162541 2.7330628
9 dias_uci 3.1969572 0.5881637 2.6087935
11 pago_hosp 0.7340365 0.0828997 0.6511369
7 quirur 0.3050877 0.0203025 0.2847852
8 dias_hosp 0.0753638 0.0314285 0.0439353
4 genero 0.0264964 0.0086772 0.0178192
5 marcas 0.0051570 0.0014848 0.0036722
6 ramo 0.0001644 0.0009926 -0.0008283
2 estrato 0.0155690 0.0179440 -0.0023750
1 edad 0.0248180 0.0273145 -0.0024965
3 est_civil 0.0112672 0.0209304 -0.0096632

De acuerdo al poder predictivo de cada una de las variables, se eligen aquellas cuyo Valor de la informaciÓn (IV) sea superior al 2% (0,02). Las variables con IV inferiores a este valor se consideran impredictivas y se decide descartarlas. Las variables que continuan, en orden de relevancia segun su poder predictor, son:

  • pago_hosp
  • quirur
  • dias_uce
  • dias_uci
  • proveedor
  • dias_hosp
  • edad
  • genero
  • ciudad

Sin embargo, tanto la ciudad, como el proveedor no serán tenidos en cuenta, por que pueden llegar a condicionar nuestra variable objetivo. Adiconalmente, lo días UCE y UCI parecen estar altamente correlacionados con la variable objetivo. por lo que tampoco serán tenidos en cuenta.

Enfocandonos en el pago del diagnóstico, el cual, es una de las variables con mayor influencia, el WOE nos indica una relación no lineal, con un incremento en el WOE a medida que disminuye el rango de pago en el diagnóstico.

pago_hosp N Percent WOE IV PENALTY
[0,107820] 2430 0.0999095 1.5471318 0.5258738 0.0273825
[108200,759828] 2433 0.1000329 -0.2229524 0.5303520 0.0335828
[760095,1531408] 2432 0.0999918 -0.0400625 0.5305094 0.0341146
[1531618,2131592] 2432 0.0999918 -0.4440218 0.5465638 0.0457588
[2132115,2895004] 2433 0.1000329 -0.5302479 0.5686062 0.0488086
[2895083,3870803] 2432 0.0999918 -0.5004264 0.5884919 0.0532475
[3870853,4944123] 2432 0.0999918 -0.9252740 0.6451388 0.0544036
[4944436,6652726] 2433 0.1000329 -0.5605176 0.6694431 0.0682488
[6653181,10266934] 2432 0.0999918 -0.4440218 0.6854975 0.0735695
[10272268,35482810] 2433 0.1000329 -0.8414772 0.7340365 0.0828997

Como se pudo observar en el analisis del WOE, esta técnica ajusta los valores de las variables numericas en rangos acotados de acuerdo al valor de la informacion de cada una de ellas en relacion con la variable dependiente. Por esto, es importante transformar dichas variables en los rangos recomendados.

Volver al Índice

Modelo

El objetivo principal del análisis es estimar un modelo predictivo con el cuál se pueda estimar la probabilidad de que un paciente termine en una rehospitalización, asociada a un diangóstico anterior. Para ello se empleará un modelo de regresión logística, el cuál es ampliamente utilizado para resolver problemas de clasificación binaria.

Una vez se realizan los filtros de calidad y completitud, y tras lo obtenido en los resultados del WOE, se procede a realizar la seleccion de variables para el modelo. Se tendrán en cuenta entonces, el pago realizado, los días en que estuvo el paciente de forma general, el hecho de que se le haya realizado o no una cirugía, el género, la edad y el estrato.

Para evaluar la capacidad de generalización del modelo, se dividirá el conjunto de datos en entrenamiento (70%) y prueba (30%).

Error in eval(lhs, parent, parent) : object 'data_rehosp' not found
Volver al Índice

Smote

Como se habia mencionado anteriormente, la informacion se encuentra desbalanceada; esto es, teniendo en cuenta que el problema en que se esta trabajando consiste en la clasificacion de una variable dicotómica, se debe analizar el nivel de representacion de sus posibles valores dentro del conjunto total de datos.

Vemos que la representacion para la categoría positiva es un poco mas del 2% de la información. En este caso vamos a realizar un tratamiento que permita aumentar la clase minoritaria, sin utilizar soluciones genéricas como reducir la clase mayoritaria al nivel de la clase menor.

Para ello, vamos a utilizar la técnica SMOTE (Synthetic Minority Oversampling Method), la cual genera nuevas instancias artificiales de la clase más pequeña interpolando los valores de las instancias minoritarias más cercanas a una dada.

Por medio de SMOTE se generará un nuevo set de datos de entrenamiento, en el cual se tenga un 60% de informacion para la categoria negativa (rehosp_oms = 0) y 40% para la categoria positiva (rehosp_oms = 0).

Verificamos que el set de entrenamiento se encuentre balanceado:

Var1 Freq
0 0.5294118
1 0.4705882
Volver al Índice

Ajuste del modelo y Estimación de parámetros

Del resultado exploratorio anterior, al discriminar el análisis de las variables independientes por nuestra variable objetivo (Rehospitalización), es posible evidenciar una diferencia clara entre las distribuciones para los atributos: Pago/costo del procedimiento y los días en que el usuario estuvo internado ya sea en la Unidad de Cuidados Intensivos o Especiales. Esto podría ser un indicio de que estas variables en particular, pueden llegar a ser relevantes para explicar la probabilidad de ocurrencia del evento, es decir, cuando hubo una hospitalización posterior ligada a un diagnóstico.

A continuación, al ajustar el modelo obtenemos los siguientes resultados:


Call:
glm(formula = rehosp_oms ~ ., family = "binomial", data = training)

Deviance Residuals: 
    Min       1Q   Median       3Q      Max  
-2.1902  -1.0617  -0.5915   1.1222   2.9389  

Coefficients:
                         Estimate Std. Error z value             Pr(>|z|)    
(Intercept)              -1.95277    0.37284  -5.238        0.00000016271 ***
pago_hosp                -0.52259    0.04475 -11.677 < 0.0000000000000002 ***
quirurSi                 -0.23013    0.06462  -3.561             0.000369 ***
dias_hosp                -0.20996    0.03807  -5.514        0.00000003499 ***
generoM                   0.28078    0.06217   4.517        0.00000628434 ***
edad31-40                 0.22532    0.11007   2.047             0.040643 *  
edad41-50                 0.14857    0.11854   1.253             0.210097    
edad51-60                 0.34300    0.11660   2.942             0.003264 ** 
edad61-70                 0.15529    0.12622   1.230             0.218606    
edad71-80                 0.57673    0.13213   4.365        0.00001272906 ***
edad81+                   0.88102    0.13655   6.452        0.00000000011 ***
estrato2                  1.09811    0.38053   2.886             0.003905 ** 
estrato3                  1.00638    0.36432   2.762             0.005738 ** 
estrato4                  0.97365    0.36386   2.676             0.007453 ** 
estrato5                  1.12560    0.36179   3.111             0.001863 ** 
estrato6                  1.01175    0.36128   2.800             0.005103 ** 
estratoSin Informacion    0.88736    0.35848   2.475             0.013311 *  
marcas(2,4]               1.07903    0.11553   9.340 < 0.0000000000000002 ***
marcas(4,6]               1.08589    0.42549   2.552             0.010709 *  
ramo79                    0.29873    0.06266   4.768        0.00000186450 ***
est_civilD                0.49587    0.15200   3.262             0.001105 ** 
est_civilS                0.26307    0.08644   3.043             0.002339 ** 
est_civilSin Informacion  0.27658    0.07703   3.591             0.000330 ***
est_civilU               -0.39354    0.16751  -2.349             0.018806 *  
est_civilV                0.58341    0.16835   3.465             0.000529 ***
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 6723.3  on 4861  degrees of freedom
Residual deviance: 6121.0  on 4837  degrees of freedom
AIC: 6171

Number of Fisher Scoring iterations: 4
  1. Cada cambio en una unidad en el pago hospitalario disminuirá las probabilidades de rehospitalización, pero en una cantidad muy pequeña (-6.953E-08)
  2. Cuando a un paciente se le realiza un procedimiento quirúrgico su probabilidad de que termine hospitalizado de nuevo por el mismo diagnóstico, disminuye en más del 19% en comparación a cuando no se le realiza ninguna cirugía.
  3. La probabilidad de rehospitalización decrece, al permanecer un día adicional el paciente en la UCE (en 1.45) y la UCI (1.73)

El resto de las variables no son suficientemente explicativas para predecir, de manera significativa, su efecto sobre la variable de respuesta binaria.

Después de estimados los coeficentes se procede a realizar la predicción dentro y fuera de muestra para evaluar la precisión (accuracy) y capacidad de generalización de nuestro modelo.

'data.frame':   4862 obs. of  10 variables:
 $ pago_hosp : num  0.138 0.129 1.064 1.163 -0.03 ...
 $ quirur    : Factor w/ 2 levels "No","Si": 1 1 1 2 1 1 2 1 2 2 ...
 $ dias_hosp : num  0.0806 -0.1694 0.0806 -0.6693 4.0799 ...
 $ genero    : Factor w/ 2 levels "F","M": 1 1 1 1 2 1 2 2 1 1 ...
 $ edad      : Factor w/ 7 levels "18-30","31-40",..: 1 6 5 6 6 7 7 4 6 1 ...
 $ estrato   : Factor w/ 7 levels "1","2","3","4",..: 6 6 4 6 7 6 5 7 6 3 ...
 $ rehosp_oms: Factor w/ 2 levels "0","1": 1 1 1 1 1 1 1 1 1 1 ...
 $ marcas    : Factor w/ 3 levels "[0,2]","(2,4]",..: 1 1 1 1 1 2 2 1 1 1 ...
 $ ramo      : Factor w/ 2 levels "26","79": 1 1 2 1 2 1 1 1 1 1 ...
 $ est_civil : Factor w/ 6 levels "C","D","S","Sin Informacion",..: 1 4 1 2 4 4 1 3 1 3 ...
Train Test
0.6458248 0.724386

Los resultados indican un nivel de accuracy sospechosamente elevado, incluso el modelo parece ajustarse mejor con los datos nuevos, que en la parte de entrenamiento. Para ver en detalle como se comporta, al discriminar entre los casos en que el paciente sale definitivamente o termina en una rehospitalización, y evidenciar su desempeño por separado, se estimará la matriz de confusión:

table(as.matrix(testing[, 7]), y_pred > 0.5)
   
    FALSE TRUE
  0  7459 2752
  1   121   92

Los resultados no parecen indicar que éste comportamiento se den a causa del desbalanceo. Por un lado tenemos que la sensibilidad y la especificidad corresponden al 96% y 99%. Si se habla en términos de precisión, sería de de un 99% cuando es 0, es decir de un total de 10218 pacientes que no terminaron en hospitalización pude predecir con una exactitud que el 99% no lo harían, mientras que de 249 pacientes que si tuvieron una rehospitalización pude predecir que 210 efectivamente lo harían, es decir, mi precisión fue del 84%.

Sin embargo, no sólo por los resultados obtenidos con el ajuste del modelo, sino también con lo que veíamos anteriormente en la tabla del valor de la información, las variables días UCI y días UCE parecen explicar de manera casi perfecta la probabilidad de que ocurre un evento rehospitalario y esto puede deberse …….. por ende se decide estimar el modelo sin incluirlas.

A continuación, se obtienen los resultados:

###Regularizado

para la regularización se utilizará el paquete glmnet, el cual asigna un valor de alpha = 0 i es ridge y alpha = 1 si lasso. Antes de sumergirse en el código, vale la pena señalar lo siguiente:

  • glmnet no tiene una interfaz de fórmula, por lo que uno tiene que ingresar los predictores como una matriz y las etiquetas de clase como un vector.

  • no acepta predictores categóricos, por lo que uno tiene que convertirlos en valores numéricos antes de pasarlos a glmnet.

La función glmnet model.matrix crea la matriz y también convierte los predictores categóricos en variables ficticias apropiadas; mientras que se usará la función cv.glmnet , para encontra de manera automáticauna el valor óptimo de lambda.

library(glmnet)
library(Matrix)
set.seed(123) 

x_train <- model.matrix(rehosp_oms~., training)[,-1]
y_train <- ifelse(training$rehosp_oms == "1", 1, 0)

##Hallando el mejor lambda
cv.lasso <- cv.glmnet(x_train, y_train, family = "binomial", type.measure = "mse", alpha = 1)
plot(cv.lasso)

La gráfica muestra que el registro del valor óptimo de lambda (es decir, el que minimiza el error cuadrático medio) es aproximadamente -3. El valor exacto se puede ver al examinar la variable lambda_min en el código a continuación. En general, sin embargo, el objetivo de la regularización es equilibrar la precisión y la simplicidad. En el contexto actual, esto significa un modelo con el menor número de coeficientes que también proporciona una buena precisión .

cv.lasso$lambda.min

En general, el propósito de la regularización es equilibrar la precisión y la simplicidad. Esto significa, un modelo con el menor número de predictores que también da una buena precisión. Para este fin, la función cv.glmnet() también encuentra el valor de lambda que proporciona el modelo más simple pero también se encuentra dentro de un error estándar del valor óptimo de lambda. Este valor se llama lambda.1se. Este valor de lambda ( lambda.1se) es lo que usaremos en el resto de la computación.

cv.lasso$lambda.1se
coef(cv.lasso, cv.lasso$lambda.min)

La salida muestra que solo aquellas variables que hemos determinado que son significativas en base a los valores de p tienen coeficientes distintos de cero, en este caso el pago hospitalario, todos los coeficientes de las demás variables han sido puestos a cero por el algoritmo.

Usando lambda.1secomo la mejor lambda, da los siguientes coeficientes de regresión:

coef(cv.lasso, cv.lasso$lambda.1se)

Usando lambda.1se, los coeficiente de 4 variables se han establecido en cero mediante el algoritmo de lazo, reduciendo la complejidad del modelo.

La configuración de lambda = lambda.1se produce un modelo más simple en comparación con lambda.min, pero el modelo podría ser un poco menos preciso que el obtenido con lambda.min.

##Lambda regression
std_ridge_logit <- glmnet(x_train, y_train, family="binomial", alpha=1)
SRL_pred_train <- predict(std_ridge_logit, x_train, type="class", s=cv.lasso$lambda.1se)

###Matriz training

confusion_matrix_train <- table(y_train, SRL_pred_train)
confusion_matrix_train
       SRL_pred_train
y_train    0    1
      0 1676  898
      1  888 1400
error_rate_train <- (760+920)/(760+920+1814+1368)
error_rate_train

###Matriz test


x_test <- model.matrix(rehosp_oms~., testing)[,-1]
y_test <- ifelse(testing$rehosp_oms == 1, 1, 0)
SRL_pred_test <- predict(std_ridge_logit, x_test, type="class", s=cv.lasso$lambda.1se)
confusion_matrix_test <- table(y_test, SRL_pred_test)
confusion_matrix_test
      SRL_pred_test
y_test    0    1
     0 6604 3607
     1   77  136
error_rate_test <- (84+2867)/(7344+2867+84+129)
error_rate_test

Calculando el modelo con ridge:

##Hallando el mejor lambda
cv.ridge <- cv.glmnet(x_train, y_train, family = "binomial", type.measure = "mse", alpha = 0)
coef(cv.ridge, cv.ridge$lambda.1se)
17 x 1 sparse Matrix of class "dgCMatrix"
                                  1
(Intercept)            -0.366246985
pago_hosp              -0.426120028
quirurSi               -0.125916773
dias_hosp              -0.104428130
generoM                 0.309270283
edad31-40              -0.037395228
edad41-50              -0.112636017
edad51-60               0.043603351
edad61-70               0.006252376
edad71-80               0.161808110
edad81+                 0.229884142
estrato2                0.231572003
estrato3                0.143485352
estrato4                0.051523729
estrato5                0.244746627
estrato6               -0.030139011
estratoSin Informacion -0.044842683
SRR_pred_test <- predict(std_ridge_logit, x_test, type="class", s=cv.ridge$lambda.1se)
confusion_matrix_test <- table(y_test, SRR_pred_test)
confusion_matrix_test
      SRR_pred_test
y_test    0    1
     0 8361 1850
     1  107  106
Volver al Índice

Recomendaciones y Estrategias

Como se pudo observar en la definicion de la variable endógena de la rehospitalización, esta se construyó mediante dos restricciones en el set de datos inicial: 1. Que dentro de los siguientes 30 dias a la primera hospitalización surgiera una segunda hospitalizacion. 2. Que para aquellos casos donde hay dos eventos en la ventana de 30 días, los códigos de diagnóstico CIE10 del primero y segundo evento pertenecieran a la misma categoría de diagóstico en la clasificación de la OMS.

En este sentido, tenemos dos condicionantes que se podrían ajustar, ya que restringen la posibilidad de encontrar un mayor numero de casos que se puedan considerar como rehospitalización; por ejemplo, algunos estudios sugieren que la ventana de tiempo podría ser de 15 dias entre el primer y segundo evento. En cuanto a la similitud de los diagnósticos de ambos eventos, para este modelo se tuvo en cuenta solo si ambos diagnósticos pertenecen a la misma categoría, sin embargo es importante tener en cuenta que hay diagnósticos que pueden desencadenar en otros que no necesariamente sean de la misma categoría. Este tipo de asociaciones requieren de un mayor análisis a nivel médico.

Se han descrito factores asociados con la rehospitalizacion relacionados con la calidad de vida de los pacientes, sin embargo en este modelo no se incluyo gran parte de este tipo de informacion, ya que, desde el principio, la población objetivo se compone de individuos asegurados en poliza de vida y salud, en su mayoria, de estrato cuatro hacia arriba. De esta forma ya no aporta información medir el nivel de calidad de vida o falicidad de acceso a los servicios de salud, pero se podria tener en cuenta información relacionada con sintomas depresivos.

La calidad en el cuidado hospitalario tambien se ha considerado como un factor importante, por lo tanto se podría considerar la agregación de información que indique que tan óptimas son las condiciones para una buena atención en los centros hospitalarios y lugares que se tuvieron en cuenta.

# PASO 1:   Carga Package y Set de datos
# ---------------------------------------------------------------------------
library(C50)
library(rpart)
library(rpart.plot) 
data(churn); # carga tablas

# PASO 2:   Crea Arbol de Decision
# ---------------------------------------------------------------------------
ModeloArbol <- rpart(rehosp_oms ~ ., data = training, parms=list(split="information"))

# PASO 3:  Predice rehospitalizacion en datos de TEST
# ---------------------------------------------------------------------------
Prediccion <- predict(ModeloArbol, testing, type="class") # Predicción en Test
MC         <- table(testing$rehosp_oms, Prediccion) # Matriz de Confusión

MC
# PASO 4: Crea Grafico
# ---------------------------------------------------------------------------
rpart.plot(ModeloArbol, type=1, extra=100,cex = .7,  box.col=c("gray99", "gray88")[ModeloArbol$frame$yval])
library(e1071)

# Ejecución del modelo SVM
modelosvm <- svm(rehosp_oms ~ ., data = training)

# Predicción de los restantes
prediccionsvm <- predict(modelosvm, new = testing)

# Tabla de confusión.
# Se usa with para que aparezca el nombre de la variable Species en ella
# ya que en caso contrario no sale.
(mc <- with(testing,(table(prediccionsvm, rehosp_oms))))
             rehosp_oms
prediccionsvm    0    1
            0 7398   99
            1 2813  114
# % correctamente clasificados
(correctos <- sum(diag(mc)) / nrow(testing) *100)
[1] 72.06447
library(ipred)

# Ejecución del modelo de Bagging
modelobag <- bagging(rehosp_oms~., data=training)

# Resumen del ajuste del modelo
#modelo
## 
## Bagging classification trees with 25 bootstrap replications 
## 
## Call: bagging.data.frame(formula = Species ~ ., data = datos.entreno)
# Hacer predicciones
prediccionbag <- predict(modelobag, testing)

# Matriz de confusión
(mc <- with(testing,table(prediccionbag, rehosp_oms)))
# Carga el paquete específico del método Random Forest
library(randomForest)

# Ajustar modelo
modeloRForest <- randomForest(rehosp_oms~., data=training)

# Resumen del ajuste del modelo
#modelo

prediccionRForest <- predict(modeloRForest, testing)

# Matriz de confusión
(mc <- with(testing, table(prediccionRForest, rehosp_oms)))
LS0tDQp0aXRsZTogIjxzcGFuIHN0eWxlPSdjb2xvcjojMzA1ZjcyJz48Y2VudGVyPjxicj5Nb2RlbG8gZGUgUHJlZGljY2nDs24gUmVob3NwaXRhbGl6YWNpw7NuPC9jZW50ZXI+Ig0Kb3V0cHV0OiBodG1sX25vdGVib29rDQpkYXRlOiAiPGNlbnRlcj5BYnJpbCAyMDE5PC9jZW50ZXI+Ig0KYXV0aG9yOiAiPGNlbnRlcj48YSBocmVmPSdtYWlsdG86YW5kcmVzLmdvbnphbGV6QGRhdGFseXRpY3MuY29tJz5BbmRyw6lzIEZlbGlwZSBHb256w6FsZXo8L2E+PC9jZW50ZXI+Ig0KLS0tDQoNCjxocj4NCjx0YWJsZT4NCjx0cj4NCjx0ZD48aW1nIHN0eWxlPSJ3aWR0aDoyODBweDsgaGVpZ2h0OjIwMHB4OyIgc3JjPSJTVVJBLnBuZyIgLz48L3RkPg0KPHRkPjxpbWcgc3R5bGU9IndpZHRoOjIwMHB4OyBoZWlnaHQ6MTUwcHg7IiBzcmM9ImJsYW5jby5qcGciIC8+PC90ZD4NCjx0ZD48aW1nIHN0eWxlPSJ3aWR0aDozNjBweDsgaGVpZ2h0OjEwMHB4OyIgc3JjPSJEYXRhbHl0aWNzLnBuZyIgLz48L3RkPg0KPC90cj4NCjwvdGFibGU+DQo8aHI+DQoNCjxoMyBpZD0iaW5kaWNlIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd249IjEiPsONbmRpY2U8L2gzPg0KPGRpdiBzdHlsZT0idGV4dC1hbGlnbjoganVzdGlmeSI+DQoNCkVsIHByZXNlbnRlIG5vdGVib29rIGNvbnRpZW5lIGluZm9ybWFjacOzbiByZWxhY2lvbmFkYSBhbCBlbnRlbmRpbWllbnRvIGRlIGxvcyBkYXRvcyB5IHJlc3VsdGFkb3Mgb2J0ZW5pZG9zIGRlIHVuIG1vZGVsbyBkZSBwcmVkaWNjacOzbiBwYXJhIGVsIHByb2JsZW1hIGRlIHJlaG9zcGl0YWxpemFjacOzbi4NCg0KPG9sPg0KICAgIDxsaT48YSBocmVmPSIjR2VuZXJhbGlkYWRlcyI+R2VuZXJhbGlkYWRlczwvYT48L2xpPg0KICAgIDxsaT48YSBocmVmPSIjRW50ZW5kaW1pZW50byI+RW50ZW5kaW1pZW50byBkZSBsb3MgZGF0b3M8L2E+PC9saT4NCiAgICA8bGk+PGEgaHJlZj0iI1BlcmRpZG9zIj5BbsOhbGlzaXMgZGUgUmVnaXN0cm9zIFDDqXJkaWRvczwvYT48L2xpPg0KICAgIDxsaT48YSBocmVmPSIjQW5hbGlzaXMiPkFuw6FsaXNpcyBFeHBsb3JhdG9yaW88L2E+PC9saT4NCiAgICAgIDxvbD4NCiAgICAgICAgPGxpPjxhIGhyZWY9IiNBbmFsaXNpc0NvbiI+QW7DoWxpc2lzIHVuaXZhcmlhZG8gLSB2YXJpYWJsZXMgY29udGludWFzPC9hPjwvbGk+DQogICAgICAgIDxsaT48YSBocmVmPSIjQW5hbGlzaXNDYXIiPkFuw6FsaXNpcyB1bml2YXJpYWRvIC0gdmFyaWFibGVzIGNhcmRpbmFsZXM8L2E+PC9saT4NCiAgICAgIDwvb2w+DQogICAgPGxpPjxhIGhyZWY9IiNBbmFsaXNpc1dPRSI+QW7DoWxpc2lzIGRlIGNsYXNpZmljYWNpw7NuIGJpbmFyaWEgdXNhbmRvIFdPRSB5IGVsIElWPC9hPjwvbGk+DQogICAgPGxpPjxhIGhyZWY9IiNNb2RlbG8iPk1vZGVsbzwvYT48L2xpPg0KICAgICAgPG9sPg0KICAgICAgICA8bGk+PGEgaHJlZj0iI1NNT1RFIj5TTU9URSAtIEJhbGFuY2VvIGRlIGNhdGVnb3JpYSBtaW5vcml0YXJpYTwvYT48L2xpPg0KICAgICAgICA8bGk+PGEgaHJlZj0iI0FqdXN0ZW1vZCI+QWp1c3RlIGRlbCBtb2RlbG8geSBFc3RpbWFjacOzbiBkZSBwYXLDoW1ldHJvczwvYT48L2xpPg0KICAgICAgICA8bGk+PGEgaHJlZj0iI0V2YWx1YWNpb24iPlByZWRpY2llbmRvIHkgZXZhbHVhbmRvIGVsIGRlc2VtcGXDsW8gZGVsIG1vZGVsbzwvYT48L2xpPg0KICAgICAgPC9vbD4NCiAgICA8bGk+PGEgaHJlZj0iI0NvbmNsdXNpb25lcyI+UmVzdWx0YWRvcyB5IENvbmNsdXNpb25lczwvYT48L2xpPg0KICAgIDxsaT48YSBocmVmPSIjUmVjb21lbmRhY2lvbiI+UmVjb21lbmRhY2lvbmVzIHkgRXN0cmF0ZWdpYXM8L2E+PC9saT4NCjwvb2w+DQoNCkhhY2llbmRvIGNsaWNrIHNvYnJlIGNhZGEgdW5hIGRlIGxhcyBzZWNjaW9uZXMgcHVlZGUgaXIgZGlyZWN0YW1lbnRlIGEgY2FkYSB1bmEgZGUgZWxsYXMuIEFsIGZpbmFsaXphciBjYWRhIHNlY2Npw7NuIGVuY29udHJhcsOhIHVuIGxpbmsgcGFyYSB2b2x2ZXIgYWwgw61uZGljZS48L2Rpdj4NCjxocj4NCg0KPGgzIGlkPSJHZW5lcmFsaWRhZGVzIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd249IjEiPkdlbmVyYWxpZGFkZXM8L2gzPg0KPGRpdiBzdHlsZT0idGV4dC1hbGlnbjoganVzdGlmeSI+DQoNCkVsIG9iamV0aXZvIGVzIGRlc2Fycm9sbGFyIGUgaW1wbGVtZW50YXIgdW4gbW9kZWxvIGRlIHByZWRpY2Npw7NuIGRlIHJlaG9zcGl0YWxpemFjaW9uZXMgcGFyYSBhcG95YXIgbG9zIHByb2dyYW1hcyBkZSBldml0YWJpbGlkYWQgcG9zdC1ob3NwaXRhbGFyaWEuIEVsIGFuw6FsaXNpcyBzZSByZWFsaXphcsOhIGNvbiBpbmZvcm1hY2nDs24gcXVlIGRlc2NyaWJlIGxhcyBjYXJhY3RlcsOtc3RpY2FzIHNvY2lvZGVtb2dyw6FmaWNhcyBkZWwgaW5kaXZpZHVvIHkgY29uIGFsZ3Vub3MgZGF0b3MgcmVjb2xlY3RhZG9zIHBvciBlbCBwZXJzb25hbCBob3NwaXRhbGFyaW87IHBhcmEgdW4gcGVyaW9kbyBkZSB0aWVtcG8gZGUgZG9zIGHDsW9zIHkgbWVkaW8sIHF1ZSB2YSBkZXNkZSAyMDE2IGhhc3RhIDIwMTguDQoNCltWb2x2ZXIgYWwgw41uZGljZV0oI2luZGljZSk8L2Rpdj4NCg0KYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUV9DQpybShsaXN0ID0gbHMoKSkNCm9wdGlvbnMoc2NpcGVuID0gMTAwKQ0KDQpzb3VyY2UoImZ1bmN0aW9uc19wbG90LlIiKQ0KDQpsaXN0Lm9mLnBhY2thZ2VzIDwtIGMoInJlYWR4bCIsICJkcGx5ciIsICJnZ3Bsb3QyIiwgImdnY29ycnBsb3QiLCAiVklNIiwgIlJDb2xvckJyZXdlciIsICJJbmZvcm1hdGlvbiIsICJrbml0ciIsICJrYWJsZUV4dHJhIiwgImdyaWRFeHRyYSIsICJza2ltciIsICJub3J0ZXN0IiwgIkdHYWxseSIsICJwbG90bHkiLCAibGF0dGljZSIsICJETXdSIiwgImNhVG9vbHMiLCAicGxvdGx5IikNCg0KbmV3LnBhY2thZ2VzIDwtIGxpc3Qub2YucGFja2FnZXNbIShsaXN0Lm9mLnBhY2thZ2VzICVpbiUgaW5zdGFsbGVkLnBhY2thZ2VzKClbLCJQYWNrYWdlIl0pXQ0KaWYobGVuZ3RoKG5ldy5wYWNrYWdlcykpIGluc3RhbGwucGFja2FnZXMobmV3LnBhY2thZ2VzKQ0KDQpsb2FkIDwtIGxhcHBseShsaXN0Lm9mLnBhY2thZ2VzLCBsaWJyYXJ5LCBjaGFyYWN0ZXIub25seSA9IFRSVUUpDQpgYGANCg0KPGhyPg0KPGgzIHN0eWxlPSJ0ZXh0LWFsaWduOiBjZW50ZXI7IiBpZD0iRW50ZW5kaW1pZW50byIgbWFya2Rvd24gPSAiMSI+RW50ZW5kaW1pZW50byBkZSBsb3MgZGF0b3M8L2gzPg0KPGRpdiBzdHlsZT0gInRleHQtYWxpZ246anVzdGlmeSI+IA0KDQpFbCBhcmNoaXZvIGNvbnRpZW5lIHJlZ2lzdHJvcyBxdWUgY29ycmVzcG9uZGVuIGEgZXZlbnRvcyBkZSByZWhvc3BpdGFsaXphY2lvbmVzIHkgc2UgZW5jdWVudHJhIGRldGFsbGFkbyBhIG5pdmVsIGRlIGNhZGEgZXZlbnRvIGhvc3BpdGFsYXJpby4gRW4gdG90YWwgc29uIDM0ODk4IHJlZ2lzdHJvcywgMTggdmFyaWFibGVzLCBkZXNjYXJ0YW5kbyBkZSBtYW5lcmEgaW5pY2lhbCwgYXF1ZWxsb3MgYXRyaWJ1dG9zIHF1ZSBzZSBkZXJpdmFuIGRlc3B1w6lzIGRlbCBzZWd1bmRvIGRpYWduw7NzdGljbzsgbG9zIGRhdG9zIHNlIGRlc2NyaWJlbiBhIGNvbnRpbnVhY2nDs246DQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KZGF0YV9yZWhvc3AgPC0gcmVhZF94bHN4KCJEQVRBX1JFSE9TUC54bHN4IixuYSA9IGMoIm5hIiwgIk5BIiwgIm51bGwiLCAiTlVMTCIpKQ0KZGF0YV9yZWhvc3AgJT4lDQogIHNlbGVjdChFZGFkX0hvc3BpdGFsaXphY2lvbiwNCiAgICAgICAgIEVzdHJhdG9fVml2aWVuZGEsDQogICAgICAgICBSYW5nb19JbmdyZXNvc19EZXNjLA0KICAgICAgICAgRXN0YWRvX0NpdmlsLA0KICAgICAgICAgR2VuZXJvLA0KICAgICAgICAgY2FudGlkYWRfbWFyY2FzLA0KICAgICAgICAgUmFtb19JZCwNCiAgICAgICAgIENpdWRhZF9Db250YWN0b19Ob21icmUsDQogICAgICAgICBDb2RpZ29fRGlhZ25vc3RpY29fT3AsDQogICAgICAgICBDYXRlZ29yaWFfRHhfSWQsDQogICAgICAgICBRdWlydXJnaWNvLA0KICAgICAgICAgRmVjaGFfSW5ncmVzb19Ib3NwLA0KICAgICAgICAgTnVtZXJvX0RpYXNfSG9zcGl0YWxhcmlvLA0KICAgICAgICAgTnVtZXJvX0RpYXNfVWNpLA0KICAgICAgICAgTnVtZXJvX0RpYXNfVWNlLA0KICAgICAgICAgUHJvdmVlZG9yLA0KICAgICAgICAgVmFsb3JfUGFnYWRvX0RpYWdub3N0aWNvLA0KICAgICAgICAgcmVob3NwX2NhdF9vbXMpICU+JQ0KICByZW5hbWUoZWRhZCA9IEVkYWRfSG9zcGl0YWxpemFjaW9uLCANCiAgICAgICAgIGVzdHJhdG8gPSBFc3RyYXRvX1ZpdmllbmRhLA0KICAgICAgICAgaW5ncmVzbyA9IFJhbmdvX0luZ3Jlc29zX0Rlc2MsDQogICAgICAgICBlc3RfY2l2aWwgPSBFc3RhZG9fQ2l2aWwsDQogICAgICAgICBnZW5lcm8gPSBHZW5lcm8sDQogICAgICAgICBtYXJjYXMgPSBjYW50aWRhZF9tYXJjYXMsDQogICAgICAgICByYW1vID0gUmFtb19JZCwNCiAgICAgICAgIGNpdWRhZCA9IENpdWRhZF9Db250YWN0b19Ob21icmUsDQogICAgICAgICBkaWFnbm9zID0gQ29kaWdvX0RpYWdub3N0aWNvX09wLA0KICAgICAgICAgY2F0ZWdvcmlhID0gQ2F0ZWdvcmlhX0R4X0lkLA0KICAgICAgICAgcXVpcnVyID0gUXVpcnVyZ2ljbywNCiAgICAgICAgIGZlY2hhX2luZ3Jlc28gPSBGZWNoYV9JbmdyZXNvX0hvc3AsDQogICAgICAgICBkaWFzX2hvc3AgPSBOdW1lcm9fRGlhc19Ib3NwaXRhbGFyaW8sDQogICAgICAgICBkaWFzX3VjaSA9IE51bWVyb19EaWFzX1VjaSwNCiAgICAgICAgIGRpYXNfdWNlID0gTnVtZXJvX0RpYXNfVWNlLA0KICAgICAgICAgcHJvdmVlZG9yID0gUHJvdmVlZG9yLA0KICAgICAgICAgcGFnb19ob3NwID0gVmFsb3JfUGFnYWRvX0RpYWdub3N0aWNvLA0KICAgICAgICAgcmVob3NwX29tcyA9IHJlaG9zcF9jYXRfb21zKSAtPiBkYXRhX3JlaG9zcA0KaGVhZChkYXRhX3JlaG9zcCkNCmBgYA0KDQo8dWw+DQo8bGk+VmFyaWFibGVzIGNvbnRpbnVhcyAoNCkNCjx1bD4NCjxsaT5kaWFzX2hvc3A6IGTDrWFzIGRlIGhvc3BpdGFsaXphY2nDs248L2xpPg0KPGxpPmRpYXNfdWNpOiBuw7ptZXJvIGTDrWFzIGVuIFVDSTwvbGk+DQo8bGk+ZGlhc191Y2U6IG7Dum1lcm8gZMOtYXMgZW4gVUNFPC9saT4NCjxsaT5wYWdvX2hvc3A6IHZhbG9yIHBhZ2FkbyBwcmltZXJhIGhvc3BpdGFsaXphY2nDs248L2xpPg0KPC91bD4NCjwvbGk+DQo8L3VsPg0KPHVsPg0KPGxpPlZhcmlhYmxlcyBub21pbmFsZXMgKDUpDQo8dWw+DQo8bGk+ZXN0cmF0bzogZXN0cmF0byBWaXZpZW5kYSAoMCwxLDIsMyw0LDUsNiwtMSk8L2xpPg0KPGxpPmVzdF9jaXZpbDogZXN0YWRvIGNpdmlsIChDLEQsUyxVLFYsLTEpPC9saT4NCjxsaT5jaXVkYWQ6IGNpdWRhZCBkZSBjb250YWN0byBkZWwgYXNlZ3VyYWRvPC9saT4NCjxsaT5kaWFnbm9zOiBjw7NkaWdvIGRpYWduw7NzdGljbyBDSUUxMCBkZSBsYSBwcmltZXJhIGF0ZW5jacOzbiA8L2xpPg0KPGxpPmNhdGVnb3JpYTogY2F0ZWdvcsOtYSBkZWwgZGlhZ27Ds3N0aWNvIHNlZ8O6biBlbCB0aXBvIGRlIGVuZmVybWVkYWQ8L2xpPg0KPC91bD4NCjwvbGk+DQo8L3VsPg0KPHVsPg0KPGxpPlZhcmlhYmxlIGRpY290w7NtaWNhICg0KQ0KPHVsPg0KPGxpPmdlbmVybzogZ8OpbmVybyBkZWwgYXNlZ3VyYWRvIChNLEYpPC9saT4NCjxsaT5yYW1vOiByYW1vIGFsIHF1ZSBwZXJ0ZW5lY2UgZWwgYXNlZ3VyYWRvPC9saT4NCjxsaT5xdWlydXI6IHNpIHR1dm8gYWxndW4gdGlwbyBkZSBzZXJ2aWNpbyByZWxhY2lvbmFkbyBhIHByb2NlZGltaWVudG8gcXVpcsO6cmdpY288L2xpPg0KPGxpPnJlaG9zcF9jYXRfb21zOiBzaW1pbGl0dWQgY2F0ZWdvcsOtYSBjaWUxMC4gRXN0YSBlcyBudWVzdHJhIHZhcmlhYmxlIG9iamV0aXZvIDwvbGk+DQo8L3VsPg0KPC9saT4NCjwvdWw+DQo8dWw+DQo8bGk+VmFyaWFibGVzIGRpc2NyZXRhcyAoMikNCjx1bD4NCjxsaT5lZGFkOiBlZGFkIGRlbCBhc2VndXJhZG8gZW4gZWwgbW9tZW50byBkZSBsYSBob3NwaXRhbGl6YWNpw7NuPC9saT4NCjxsaT5tYXJjYXM6IGNhbnRpZGFkIGRlIG1hcmNhcyBjb25maXJtYWRhcyBkZWwgYXNlZ3VyYWRvPC9saT4NCjwvdWw+DQo8L2xpPg0KPC91bD4NCjx1bD4NCjxsaT5WYXJpYWJsZXMgb3JkaW5hbGVzICgxKQ0KPHVsPg0KPGxpPmluZ3Jlc286IHJhbmdvIGRlIGluZ3Jlc29zPC9saT4NCjwvdWw+DQo8L2xpPg0KPC91bD4NCjx1bD4NCjxsaT5GZWNoYSAoMSkNCjx1bD4NCjxsaT5GZWNoYV9JbmdyZXNvOiBmZWNoYSBpbmdyZXNvIGhvc3BpdGFsaXphY2nDs24gPC9saT4NCjwvdWw+DQo8L2xpPg0KPC91bD4NCg0KR2VuZXJhbW9zIGxhIGVzdGFkw61zdGljYSBkZXNjcmlwdGl2YTsgZW4gZWxsYSBzZSBwdWVkZSB2aXN1YWxpemFyIHF1ZSBzZXLDoSBuZWNlc2FyaW8gcmVhbGl6YXIgbcOhcyBhZGVsYW50ZSBhbGd1bmFzIGNvbnZlcnNpb25lcyBlbiBsb3MgdGlwb3MgZGUgZGF0b3MgcXVlIHZpZW5lbiBwb3IgZGVmZWN0byAocG9yIGVqZW1wbG8gZWwgZXN0cmF0byBhcGFyZWNlIGNvbW8gdW5hIHZhcmlhYmxlIG51bcOpcmljYSkuIFBlcm8gYW50ZXMgZGUgY29udGludWFyIGNvbiBsYSBjb2RpZmljYWNpw7NuLCBwcm9jZWRlcmVtb3MgYSBvYnNlcnZhciBjb21vIHNlIGVuY3VlbnRyYW4gbnVlc3RyYXMgdmFyaWFibGVzLg0KDQo8YnI+DQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0Kc2tpbV93aXRoKG51bWVyaWMgPSBsaXN0KGhpc3QgPSBOVUxMKSkNCg0KZGF0YV9yZWhvc3AgJT4lIA0KICBncm91cF9ieSgpICU+JQ0KICBza2ltKCkNCmBgYA0KDQo8dWw+DQo8bGk+TGEgdmFyaWFibGUgaW5ncmVzbyBlcyBsYSBxdWUgbcOhcyBkYXRvcyBwZXJkaWRvcyB0aWVuZSwgc2VndWlkYSBkZWwgZXN0cmF0by4NCjxsaT5MYSBtYXlvcsOtYSBkZSBsb3MgcGFjaWVudGVzIG5vIGVzdHV2aWVyb24gaW5ncmVzYWRvcyBlbiBlbCBVQ0kgbyBlbiBVQ0UNCjxsaT5FbCBwYWdvIGhvc3BpdGFsYXJpbyBwcm9tZWRpbyBmdWUgZGUgJDUnNzA2LDEwOCBjb24gdW5hIGRlc3ZpYWNpw7NuIGRlICQxMSc0NDcsMTM1IGRlIGxhIG1lZGlhLCBsbyBxdWUgaW5kaWNhIHVuYSBncmFuIGRpc3BlcnNpw7NuIGVuIGxvcyBkYXRvcywgeSBwb3NpYmxlIHByZXNlbmNpYSBkZSBvdXRsaWVycy4NCjxsaT5FbCA3NSUgZGUgbGEgaW5mb3JtYWNpw7NuIGNvcnJlc3BvbmRlIGEgdXN1YXJpb3MgcXVlIG5vIHNlIHJlaG9zcGl0YWxpemFyb24sIHBvciBlbmRlIG51ZXN0cmEgdmFyaWFibGUgb2JqZXRpdm8gc2UgZW5jdWVudHJhIGRlc2JhbGFuY2VhZGENCjx1bD4NCg0KPGJyPg0KDQpBIGNvbnRpbnVhY2lvbiB2ZXJlbW9zIGFsZ3VuYXMgZ3JhZmljYXMgcXVlIHBlcm1pdGVuIHJlYWxpemFyIGluZmVyZW5jaWFzIGFjZXJjYSBkZWwgY29tcG9ydGFtaWVudG8gZGUgbG9zIGRhdG9zOg0KYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUUsIGZpZy53aWR0aD04fQ0KcmVxdWlyZShzY2FsZXMpDQoNCmRhdGFfcmVob3NwICU+JQ0KICBmaWx0ZXIocGFnb19ob3NwID4gMCAmIHF1aXJ1ciA9PSAwICYgcmVob3NwX29tcyA9PSAxKSAlPiUNCiAgZ3JvdXBfYnkoZmVjaGFfaW5ncmVzbykgJT4lDQogIHN1bW1hcmlzZV9hbGwofnN1bShwYWdvX2hvc3ApKSAlPiUNCiAgICBnZ3Bsb3QoDQogICAgICBhZXMoDQogICAgICAgIHg9ZmVjaGFfaW5ncmVzbywgDQogICAgICAgIHk9cGFnb19ob3NwKSkgKw0KICAgIGdlb21fbGluZSgNCiAgICAgIGNvbG9yID0gIiM5OUNDRkYiLCANCiAgICAgIHNpemUgPSAwLjMpICsgDQogICAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIikgKyANCiAgICB0aGVtZV9taW5pbWFsKCkgKw0KICAgIGxhYnModGl0bGUgPSAiUGFnbyBIb3NwIGVuIGVsIHRpZW1wbyIsDQogICAgICAgICB4PSAiRmVjaGEgSW5ncmVzbyIsDQogICAgICAgICB5ID0gIlBhZ28gSG9zcCIpICsNCiAgICB0aGVtZShwbG90LnRpdGxlID0gZWxlbWVudF90ZXh0KGhqdXN0ID0gMC41KSwNCiAgICAgICAgICBsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpICsgDQogICAgc2NhbGVfeV9jb250aW51b3VzKGxhYmVscyA9IGRvbGxhcikgLT4gcDExDQoNCnAxMSA8LSBnZ3Bsb3RseShwMTEpDQpwMTENCmBgYA0KDQpgYGB7cn0NCmRhdGFfcmVob3NwICU+JQ0KICBncm91cF9ieShjaXVkYWQpICU+JQ0KICBzdW1tYXJpc2VfYWxsKH5zdW0ocmVob3NwX29tcykpICU+JQ0KICBmaWx0ZXIocmVob3NwX29tcyA+IDgpICU+JQ0KICAgIGdncGxvdCgNCiAgICAgIGFlcyhmaWxsPWNpdWRhZCwNCiAgICAgICAgICB4PWNpdWRhZCwgDQogICAgICAgICAgeT1yZWhvc3Bfb21zKSkgKw0KICAgIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiKSArIA0KICAgIHRoZW1lX21pbmltYWwoKSArDQogICAgbGFicyh0aXRsZSA9ICJSZWhvc3BpdGFsaXphY2lvbiBwb3IgY2l1ZGFkIiwNCiAgICAgICAgIHg9ICJDaXVkYWQiLA0KICAgICAgICAgeSA9ICJDYW50LiBSZWhvc3AiKSArDQogICAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA5MCksDQogICAgICAgICAgcGxvdC50aXRsZSA9IGVsZW1lbnRfdGV4dChoanVzdCA9IDAuNSksDQogICAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKSArDQogICAgc2NhbGVfZmlsbF9icmV3ZXIocGFsZXR0ZSA9ICJCbHVlcyIpLT4gcDEyDQoNCnAxMiA8LSBnZ3Bsb3RseShwMTIpDQpwMTINCg0KYGBgDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFLCBmaWcud2lkdGg9OCwgZmlnLmhlaWdodD00fQ0KIHJlcXVpcmUoc2NhbGVzKQ0KIA0KZGF0YV9yZWhvc3AgJT4lDQogIGZpbHRlcihyZWhvc3Bfb21zID4gMCkgJT4lDQogIGdyb3VwX2J5KGZlY2hhX2luZ3Jlc28pICU+JQ0KICBzdW1tYXJpc2VfYWxsKH5zdW0ocmVob3NwX29tcykpICU+JQ0KICAgIGdncGxvdCgNCiAgICAgIGFlcygNCiAgICAgICAgeD1mZWNoYV9pbmdyZXNvLCANCiAgICAgICAgeT1yZWhvc3Bfb21zKSkgKw0KICAgIGdlb21fcG9pbnQoY29sb3I9IiM5OUNDRkYiLCANCiAgICAgIHNpemU9MikgKyANCiAgICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iKSArIA0KICAgIHRoZW1lX21pbmltYWwoKSArDQogICAgbGFicyh0aXRsZSA9ICJOdW1lcm8gUmVob3NwIGVuIGVsIHRpZW1wbyIsDQogICAgICAgIHg9ICJGZWNoYSBSZWhvc3AuIiwNCiAgICAgICAgeSA9ICJOdW0uIFJlaG9zcC4iKSArDQogICB0aGVtZShwbG90LnRpdGxlID0gZWxlbWVudF90ZXh0KGhqdXN0ID0gMC41KSwNCiAgICAgICAgIGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgLT4gcDEzDQogDQpwMTMgPC0gZ2dwbG90bHkocDEzKQ0KcDEzDQpgYGANCg0KW1ZvbHZlciBhbCDDjW5kaWNlXSgjaW5kaWNlKTwvZGl2Pg0KDQo8aHI+DQo8aDIgaWQ9IlBlcmRpZG9zIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd249IjEiPkdlbmVyYWxpZGFkZXM8L2gyPg0KPGRpdiBzdHlsZT0idGV4dC1hbGlnbjoganVzdGlmeSI+DQoNCkVuIGxhIGdyw6FmaWNhIHNpZ3VpZW50ZSBwb2RlbW9zIG9ic2VydmFyIHF1ZSBoYXkgZW4gdG90YWwgNCB2YXJpYWJsZXMgcXVlIGNvbnRpZW5lbiByZWdpc3Ryb3MgdmFjaW9zOiBlc3RyYXRvLCBlc3RhZG8gY2l2aWwsIGluZ3Jlc28geSBwcm92ZWVkb3IuDQoNCkEgbml2ZWwgaW5kaXZpZHVhbCBlbCBwb3JjZW50YWplIGRlIHZhbG9yZXMgcGVyZGlkb3MgcGFyYSB0b2RvcyBsb3MgY2Fzb3MgZXMgc3VwZXJpb3IgYWwgMjUlLiBEZSBmb3JtYSBjb21iaW5hZGEgaGF5IG3DoXMgZGVsIDMwJSBkZSBjYW1wb3MgdmFjw61vczsgcG9yIGVuZGUgbm8gcG9kZW1vcyBkZWNpciBxdWUgbGEgcHJvYmFiaWxpZGFkIGRlIHF1ZSBmYWx0ZSB1biB2YWxvciBkZXBlbmRlIHNvbG8gZGVsIHZhbG9yIG9ic2VydmFkbywgeSB1c2FyIHVuIG3DqXRvZG8gcGFyYSBpbXB1dGFybG8gKGxhIGZvcm1hIG5vIGVzIGFsZWF0b3JpYSkuDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZXZhbCA9VFJVRSwgZWNobyA9IEZBTFNFfQ0KYWdncihkYXRhX3JlaG9zcCwNCiAgICAgY29sID0gYygiI0NDRTVGRiIsICIjOTlDQ0ZGIiksDQogICAgIGNleC5heGlzID0gMC43LA0KICAgICBwcm9wID0gYyhUUlVFLCBGQUxTRSksDQogICAgIG51bWJlciA9IFRSVUUsDQogICAgIGdhcCA9IDEuNSwNCiAgICAgYm9yZGVyID0gTkEsDQogICAgIGJhcnMgPSBGQUxTRSwNCiAgICAgeWxhYiA9IGMoIlByb3BvcmNpw7NuIGRlIERhdG9zIFBlcmRpZG9zIiwgIkNvbWJpbmFjaW9uZXMiKSkNCmBgYA0KDQpEYWRvIGxvIGFudGVyaW9yLCBzZSBoYWNlIG5lY2VzYXJpbyBjb25zdHJ1aXIgdW5hIHRlcmNlcmEgY2F0ZWdvcsOtYSwgcG9yIGxvIG1lbm9zIHBhcmEgbGFzIHZhcmlhYmxlcyBxdWUgcG9zZWVuIG1lbm9zIGNhbXBvcyB2YWPDrW9zIChlc3RyYXRvIHkgZXN0YWRvIGNpdmlsKS4NCg0KUGFyYSBlc3RpbWFyIHNpIGV4aXN0ZSB1bmEgYXNvY2lhY2nDs24gZW50cmUgbGFzIHZhcmlhYmxlcyBxdWUgcHVlZGEgZGVyaXZhcnNlIGVuIGNvbGluZWFsaWRhZCwgc2UgcHJvY2VkZSBwcmltZXJvIGEgdmVyaWZpY2FyIHF1ZSBsYXMgdmFyaWFibGVzIG5vIHBvc2VlbiB1bmEgZGlzdHJpYnVjacOzbiBub3JtYWwsIHVuYSB2ZXogcmVhbGl6YWRvIGVzdG8sIHNlIGVsaWdlIGVsIHRlc3QgZGUgU3BlYXJtYW4gcGFyYSBoYWxsYXIgbGEgY29ycmVsYWNpw7NuIGxpbmVhbCBwb3IgYXRyaWJ1dG8uDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KZGF0YV9yZWhvc3AgJT4lDQogIHNlbGVjdChwYWdvX2hvc3AsIA0KICAgICAgICAgZGlhc191Y2ksDQogICAgICAgICBkaWFzX3VjZSwNCiAgICAgICAgIGRpYXNfaG9zcCwNCiAgICAgICAgIHJlaG9zcF9vbXMpIC0+IGRhdGFfbnVtDQoNCm5vcm1fdGVzdCA8LSBsYXBwbHkoZGF0YV9udW0sIGxpbGxpZS50ZXN0KQ0KbHJlcyA8LSBzYXBwbHkobm9ybV90ZXN0LCBgW2AsIGMoInN0YXRpc3RpYyIsInAudmFsdWUiKSkNCnQobHJlcykNCmBgYA0KDQpMb3MgcmVzdWx0YWRvcyBjb25maXJtYW4gcXVlIG5pbmd1bmEgZGUgbGFzIHZhcmlhYmxlcyBwZXNlbnRhIHVuYSBkaXN0cmlidWNpw7NuIG5vcm1hbCB5IGxhcyBjb3JyZWxhY2lvbmVzIHJlbGFjaW9uYWRhcyBhIGNvbnRpbnVhY2nDs24sIHZlcmlmaWNhbiBwb3NpYmxlcyBhc29jaWFjaW9uZXMgZW50cmUgbGFzIHZhcmlhYmxlcyBkZSBsb3MgZMOtYXMgZW4gcXVlIGVsIHBhY2llbnRlIGVzdHV2byBpbnRlcm5hZG8gZW4gbGEgVW5pZGFkIGRlIEN1aWRhZG9zIEludGVuc2l2b3MsIGVuIGxhIFVuaWRhZCBkZSBDdWlkYWRvcyBFc3BlY2lhbGVzIHkgbG9zIGTDrWFzIHF1ZSBlbCBwYWNpZW50ZSBlc3R1dm8gaG9zcGl0YWxpemFkby4gUG9yIGNvbm9jaW1pZW50byBkZSBmYWN0bywgbGEgcmVsYWNpw7NuIGVudHJlIGxhIHZhcmlhYmxlICJkaWFzX3VjaSIgeSAiZGlhc191Y2UiIGVzIGVudGVuZGlibGUsIHlhIHF1ZSBjdWFuZG8gdW4gcGFjaWVudGUgcXVlIGhhIHBhc2FkbyBwb3IgbGEgVW5pZGFkIGRlIEN1aWRhZG9zIEludGVuc2l2b3MgcGFzw7Mgc3UgbW9tZW50byBkZSBjcmlzaXMgeSBzdSBlc3RhZG8gZGUgc2FsdWQgZXMgbcOhcyBlc3RhYmxlLCBzdWVsZSBzZXIgcmVtaXRpZG8gYSBsYSBVbmlkYWQgZGUgQ3VpZGFkb3MgRXNwZWNpYWxlcy4gDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KY29ycl9udW0gPC0gcm91bmQoY29yKGRhdGFfbnVtKSw0KQ0KcC5tYXQgPC0gY29yX3BtYXQoZGF0YV9udW0sIG1ldGhvZCA9ICJzcGVhcm1hbiIpDQoNCmdnY29ycnBsb3QoY29ycl9udW0sIA0KICAgICAgICAgICB0eXBlID0gImxvd2VyIiwNCiAgICAgICAgICAgb3V0bGluZS5jb2wgPSAid2hpdGUiLA0KICAgICAgICAgICBwLm1hdCA9IHAubWF0LA0KICAgICAgICAgICBzaWcubGV2ZWwgPSAwLjA1LA0KICAgICAgICAgICBnZ3RoZW1lID0gZ2dwbG90Mjo6dGhlbWVfbWluaW1hbCwNCiAgICAgICAgICAgbGFiID0gVFJVRSwNCiAgICAgICAgICAgY29sb3JzID0gYygiIzk5Q0NGRiIsICJ3aGl0ZSIsICIjMDA2NkNDIikpICsgDQogIGxhYnModGl0bGUgPSAiQ29ycmVsYWNpb24gZW50cmUgdmFyaWFibGVzIG51bcOpcmljYXMiKQ0KYGBgDQoNClNpbiBlbWJhcmdvLCBsYXMgY29ycmVsYWNpb25lcyBvYnRlbmlkYXMgbm8gY3VtcGxlbiB1biB1bWJyYWwgc3VmaWNpZW50ZSBwYXJhIGNvbnNpZGVyYXJsYXMgaW1wb3J0YW50ZXMsIHBvciBlbmRlIHNlIHByb2NlZGUgYSBjb25zZXJ2YXJsYXMgeSBldmFsdWFyIG3DoXMgYWRlbGFudGUgc2kgZXMgcHJlY2lzbyBlbGltaW5hcmxhcyBkZWZpbml0aXZhbWVudGUuIFBvciBvdHJvIGxhZG8sIGxhIHZhcmlhYmxlIGNhdGVnb3LDrWEgeSBkaWFnbsOzc3RpY28gZXN0w6FuIGFsdGFtZW50ZSBjb3JyZWxhY2lvbmFkYXMgY29uIGxhIHZhcmlhYmxlIGVuZMOzZ2VuYSwgcG9yIGxvIHF1ZSBlcyBuZWNlc2FyaW8gZWxpbWluYXJsYXMgZGVsIGFuw6FsaXNpcywgcGFyYSBubyBpbmN1cnJpciBlbiBwb3NpYmxlcyBzb2JyZWFqdXN0ZXMgZW4gbGEgZXRhcGEgZGUgbW9kZWxhZG8uDQoNCkNvbW8gc2UgaGFiw61hIG1lbmNpb25hZG8gYW50ZXJpb3JtZW50ZSwgdGVuaWVuZG8gZW4gY3VlbnRhIGVsIGFuw6FsaXNpcyBkZSBkYXRvcyBwZXJkaWRvcyBvIG51bG9zLCBzZSBkZWNpZGUgZGVzY2FydGFyIGxhIHZhcmlhYmxlIGluZ3Jlc28geWEgcXVlIGNvbnRpZW5lIG1hcyBkZSB1biAzMCUgZW4gZGF0b3MgcGVyZGlkb3MuDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KDQpkYXRhX3JlaG9zcCAlPiUNCiAgbXV0YXRlKGVzdHJhdG8gPSBpZmVsc2UoaXMubmEoZXN0cmF0bykgfCBlc3RyYXRvID09IC0xIHwgZXN0cmF0byA9PSAwLCAiU2luIEluZm9ybWFjaW9uIiwgZXN0cmF0byksDQogICAgICAgICBlc3RfY2l2aWwgPSBpZmVsc2UoaXMubmEoZXN0X2NpdmlsKSwgIlNpbiBJbmZvcm1hY2lvbiIsIGVzdF9jaXZpbCksDQogICAgICAgICBpbmdyZXNvID0gaWZlbHNlKGlzLm5hKGluZ3Jlc28pLCAiU2luIEluZm9ybWFjaW9uIiwgaW5ncmVzbyksDQogICAgICAgICBwcm92ZWVkb3IgPSBpZmVsc2UoaXMubmEocHJvdmVlZG9yKSwgIlNpbiBJbmZvcm1hY2lvbiIsIHByb3ZlZWRvciksDQogICAgICAgICBxdWlydXIgPSBpZmVsc2UocXVpcnVyID09IDEsICdTaScsICdObycpLA0KICAgICAgICAgZWRhZCA9IGNhc2Vfd2hlbiggZWRhZCA8PSAzMCB+ICIxOC0zMCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDMxICYgZWRhZCA8PSA0MCB+ICIzMS00MCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDQxICYgZWRhZCA8PSA1MCB+ICI0MS01MCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDUxICYgZWRhZCA8PSA2MCB+ICI1MS02MCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDYxICYgZWRhZCA8PSA3MCB+ICI2MS03MCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDcxICYgZWRhZCA8PSA4MCB+ICI3MS04MCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICBlZGFkID49IDgxIH4gIjgxKyIpLA0KICAgICAgICAgbWFyY2FzID0gY3V0KG1hcmNhcywgYnJlYWtzID0gKDA6MykqMiwgaW5jbHVkZS5sb3dlc3QgPSBUUlVFKSwNCiAgICAgICAgIGVzdF9jaXZpbCA9IGFzLmZhY3Rvcihlc3RfY2l2aWwpLA0KICAgICAgICAgZ2VuZXJvID0gYXMuZmFjdG9yKGdlbmVybyksDQogICAgICAgICBxdWlydXIgPSBhcy5mYWN0b3IocXVpcnVyKSwNCiAgICAgICAgIHJhbW8gPSBhcy5mYWN0b3IocmFtbyksDQogICAgICAgICBlZGFkID0gYXMuZmFjdG9yKGVkYWQpLA0KICAgICAgICAgZXN0cmF0byA9IGFzLmZhY3Rvcihlc3RyYXRvKSkgJT4lDQogIHNlbGVjdCgtZGlhZ25vcywgLWNhdGVnb3JpYSwgLWZlY2hhX2luZ3Jlc28sIC1pbmdyZXNvLCAtcHJvdmVlZG9yLCAtY2l1ZGFkKSAtPiBkYXRhX3JlaG9zcA0KDQpzdHIoZGF0YV9yZWhvc3ApDQpgYGANCg0KW1ZvbHZlciBhbCDDjW5kaWNlXSgjaW5kaWNlKTwvZGl2Pg0KDQo8aHI+DQo8aDIgaWQ9IkFuYWxpc2lzIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd249IjEiPkFuw6FsaXNpcyBFeHBsb3JhdG9yaW88L2gyPg0KDQo8aDMgaWQ9IkFuYWxpc2lzQ29uIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyJtYXJrZG93bj0iMSI+QW7DoWxpc2lzIHVuaXZhcmlhZG8gLSB2YXJpYWJsZXMgY29udGludWFzPC9oMz4NCjxkaXYgc3R5bGU9InRleHQtYWxpZ246IGp1c3RpZnkiPg0KDQpFcyBldmlkZW50ZSBsYSBleGlzdGVuY2lhIHRhbWJpw6luLCBkZSB2YWxvcmVzIGF0w61waWNvcyBtdXkgbWFyY2Fkb3MgdGFudG8gZW4gZWwgbnVtw6lybyBkZSBkw61hcyBkZSBob3NwaXRhbGl6YWNpw7NuLCBjb21vIGVuIGxvcyBuw7ptZXJvcyBkZSBkw61hcyBxdWUgZWwgcGFjaWVudGUgZXN0dXZvIGVuIGxhIFVuaWRhZCBkZSBDdWlkYWRvIEludGVuc2l2byB5IEVzcGVjaWFsLCBlbiBkw7NuZGUgbG9zIHZhbG9yZXMgYXTDrXBpY29zIG3DoXMgZ3JhbmRlcyBzdWNlZGVuIGVuIGxvcyBldmVudG9zIHF1ZSBubyBkZXNlbmNhZGVuYXJvbiBlbiByZWhvc3BpdGFsaXphY2nDs24uDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KDQpkYXRhX3JlaG9zcCAlPiUNCiAgZmlsdGVyKGRpYXNfdWNpID4gMCkgJT4lDQogIGdncGxvdCAoLiwgYWVzICggZGlhc191Y2ksIGNvbG9yID0gYXMuZmFjdG9yKHJlaG9zcF9vbXMpKSkgKw0KICBnZW9tX2hpc3RvZ3JhbShmaWxsPSJ3aGl0ZSIsIGFscGhhPTAuNSwgcG9zaXRpb249ImlkZW50aXR5IikgKw0KICB0aGVtZV9taW5pbWFsKCkgKw0KICBsYWJzKHggPSAiRMOtYXMgVUNJIiwgDQogICAgICAgeSA9ICJGcmVjdWVuY2lhIFJlbGF0aXZhIiwgDQogICAgICAgZmlsbCA9ICIgIikgKw0KICBnZ3RpdGxlKCJEw61hcyBlbiBVbmlkYWQgZGUgQ3VpZGFkb3MiKSArDQogIHRoZW1lKHBsb3QudGl0bGUgPSBlbGVtZW50X3RleHQoaGp1c3QgPSAwLjUpKSArIA0KICAgIHNjYWxlX2NvbG91cl9icmV3ZXIocGFsZXR0ZSA9ICJQYWlyZWQiKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgLT4gcDENCg0KZGF0YV9yZWhvc3AgJT4lDQogIGZpbHRlcihkaWFzX3VjZSA+IDApICU+JQ0KICBnZ3Bsb3QgKC4sIGFlcyAoIGRpYXNfdWNlLCBjb2xvciA9IGFzLmZhY3RvcihyZWhvc3Bfb21zKSkpICsNCiAgZ2VvbV9oaXN0b2dyYW0oZmlsbD0id2hpdGUiLCBhbHBoYT0wLjUsIHBvc2l0aW9uPSJpZGVudGl0eSIpICsNCiAgdGhlbWVfbWluaW1hbCgpICsNCiAgbGFicyh4ID0gIkTDrWFzIFVDRSIsIA0KICAgICAgIHkgPSAiRnJlY3VlbmNpYSBSZWxhdGl2YSIsIA0KICAgICAgIGZpbGwgPSAiUmVob3NwaXRhbGl6YWNpw7NuIikgKw0KICBnZ3RpdGxlKCJOw7ptZXJvIGRlIGTDrWFzIGVuIFVuaWRhZCBkZSBDdWlkYWRvcyIpICsNCiAgdGhlbWUocGxvdC50aXRsZSA9IGVsZW1lbnRfdGV4dChoanVzdCA9IDAuNSkpICsgDQogICAgc2NhbGVfY29sb3VyX2JyZXdlcihwYWxldHRlID0gIlBhaXJlZCIpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikgLT4gcDINCg0KcDEgPC0gZ2dwbG90bHkocDEpDQpwMiA8LSBnZ3Bsb3RseShwMikNCg0Kc3VicGxvdChwMSwgcDIsIHRpdGxlWCA9IFRSVUUsIHRpdGxlWSA9IFRSVUUpICU+JQ0KICBsYXlvdXQoc2hvd2xlZ2VuZCA9IChGQUxTRSkpDQpgYGANCg0KQ29uIGVsIGFuw6FsaXNpcyBhbnRlcmlvciBubyBzw7NsbyBzZSBsb2dyYSBpZGVudGlmaWNhciBsYSBwcmVzZW5jaWEgZGUgdmFsb3JlcyBhdMOtcGljb3MsIHNpbm8gcXVlIHRhbWJpw6luIGVzIHBvc2libGUgZXZpZGVuY2lhciBxdWUgbG9zIGRhdG9zIHNlIGVuY3VlbnRyYW4gYWx0YW1lbnRlIGRlc2JhbGFuY2VhZG9zLiBFbiBlbCBjYXNvIGRlIGxvcyBvdXRsaWVycyBzZSB0cnVuY2Fyw6EgZW4gbG9zIGNhc29zIGVuIHF1ZSBzZWEgbmVjZXNhcmlvLCBpbXB1dGFuZG8gbG9zIHZhbG9yZXMgcXVlIHN1cGVyZW4gY2llcnRvIGzDrW1pdGUgZW4gZWwgcGVyY2VudGlsLCB0YW50byBtYXlvciBjb21vIG1lbm9yLg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCmRhdGFfcmVob3NwICU+JQ0KICBtdXRhdGUocGFnb19ob3NwID0gaWZlbHNlKHF1aXJ1ciA9PSAiU2kiICYgcmVob3NwX29tcyA9PSAwLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdXRsaWVyKG15ZGF0YSA9IGZpbHRlcihkYXRhX3JlaG9zcCwgcXVpcnVyID09ICJTaSIgJiByZWhvc3Bfb21zID09IDApLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gInBhZ29faG9zcCIsIHFfbWluID0gMCwgcV9tYXggPSAwLjk3KSwgcGFnb19ob3NwKSwNCiAgICAgICAgIHBhZ29faG9zcCA9IGlmZWxzZShxdWlydXIgPT0gIlNpIiAmIHJlaG9zcF9vbXMgPT0gMSwgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0bGllcihteWRhdGEgPSBmaWx0ZXIoZGF0YV9yZWhvc3AsIHF1aXJ1ciA9PSAiU2kiICYgcmVob3NwX29tcyA9PSAxKSwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gInBhZ29faG9zcCIsIHFfbWluID0gMCwgcV9tYXggPSAwLjk4KSwgcGFnb19ob3NwKSwNCiAgICAgICAgIHBhZ29faG9zcCA9IGlmZWxzZShxdWlydXIgPT0gIk5vIiAmIHJlaG9zcF9vbXMgPT0gMCwgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0bGllcihteWRhdGEgPSBmaWx0ZXIoZGF0YV9yZWhvc3AsIHF1aXJ1ciA9PSAiTm8iICYgcmVob3NwX29tcyA9PSAwKSwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9ICJwYWdvX2hvc3AiLCBxX21pbiA9IDAsIHFfbWF4ID0gMC45NyksIHBhZ29faG9zcCksIA0KICAgICAgICAgcGFnb19ob3NwID0gaWZlbHNlKHF1aXJ1ciA9PSAiTm8iICYgcmVob3NwX29tcyA9PSAxLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdXRsaWVyKG15ZGF0YSA9IGZpbHRlcihkYXRhX3JlaG9zcCwgcXVpcnVyID09ICJObyIgJiByZWhvc3Bfb21zID09IDEpLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gInBhZ29faG9zcCIsIHFfbWluID0gMCwgcV9tYXggPSAwLjk4KSwgcGFnb19ob3NwKSwNCiAgICAgICAgIGRpYXNfaG9zcCA9IGlmZWxzZShyZWhvc3Bfb21zID09IDAsIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgIG91dGxpZXIobXlkYXRhID0gZmlsdGVyKGRhdGFfcmVob3NwLCByZWhvc3Bfb21zID09IDApLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9ICJkaWFzX2hvc3AiLCBxX21pbiA9IDAsIHFfbWF4ID0gMC45OSksDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0bGllcihteWRhdGEgPSBmaWx0ZXIoZGF0YV9yZWhvc3AsIHJlaG9zcF9vbXMgPT0gMSksIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gImRpYXNfaG9zcCIsIHFfbWluID0gMCwgcV9tYXggPSAwLjk5KSksDQogICAgICAgICBkaWFzX3VjaSA9IGlmZWxzZShyZWhvc3Bfb21zID09IDAsIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0bGllcihteWRhdGEgPSBmaWx0ZXIoZGF0YV9yZWhvc3AsIHJlaG9zcF9vbXMgPT0gMCAmIGRpYXNfdWNpID4gMCksIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gImRpYXNfdWNpIiwgcV9taW4gPSAwLCBxX21heCA9IDAuOTk5KQ0KICAgICAgICAgICAgICAgICAgICAgICAgICAgLCBkaWFzX3VjaSksDQogICAgICAgICBkaWFzX3VjZSA9IGlmZWxzZShyZWhvc3Bfb21zID09IDAsIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0bGllcihteWRhdGEgPSBmaWx0ZXIoZGF0YV9yZWhvc3AsIHJlaG9zcF9vbXMgPT0gMCAmIGRpYXNfdWNlID4gMCksIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gImRpYXNfdWNlIiwgcV9taW4gPSAwLCBxX21heCA9IDAuOTk5KQ0KICAgICAgICAgICAgICAgICAgICAgICAgICAgLCBkaWFzX3VjaSkpIC0+IGRhdGFfcmVob3NwDQpgYGANCg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCg0KcDMgPC0gbXlib3hwbG90KG15ZGF0YSA9IGZpbHRlcihkYXRhX3JlaG9zcCwgcGFnb19ob3NwID4gMCksIA0KICAgICAgICAgICAgICAgIG15ZXhwb3N1cmUgPSAicmVob3NwX29tcyIsIA0KICAgICAgICAgICAgICAgIG15b3V0Y29tZSA9ICJwYWdvX2hvc3AiLCANCiAgICAgICAgICAgICAgICBteXRpdGxlID0gIlBhZ28gaG9zcGl0YWxpemFjacOzbiIsIA0KICAgICAgICAgICAgICAgIG15bGFiZWxfeCA9ICIiLCANCiAgICAgICAgICAgICAgICBteWxhYmVsX3kgPSAiUGFnbyBEaWFnIiwgDQogICAgICAgICAgICAgICAgbXlfZmlsbCA9ICIiKQ0KDQpwNCA8LSBteWJveHBsb3QobXlkYXRhID0gZGF0YV9yZWhvc3AsIA0KICAgICAgICAgICAgICAgIG15ZXhwb3N1cmUgPSAicmVob3NwX29tcyIsIA0KICAgICAgICAgICAgICAgIG15b3V0Y29tZSA9ICJkaWFzX2hvc3AiLCANCiAgICAgICAgICAgICAgICBteXRpdGxlID0gIlRvdGFsIGRpYXMgaG9zcGl0YWxpemFkbyIsIA0KICAgICAgICAgICAgICAgIG15bGFiZWxfeCA9ICIiLCANCiAgICAgICAgICAgICAgICBteWxhYmVsX3kgPSAiRGlhcyBob3NwaXRhbGl6YWNpw7NuIiwgDQogICAgICAgICAgICAgICAgbXlfZmlsbCA9ICIiKQ0KDQpwNSA8LSBteWJveHBsb3QobXlkYXRhID0gZmlsdGVyKGRhdGFfcmVob3NwLCBkaWFzX3VjaSA+IDApLCANCiAgICAgICAgICAgICAgICBteWV4cG9zdXJlID0gInJlaG9zcF9vbXMiLCANCiAgICAgICAgICAgICAgICBteW91dGNvbWUgPSAiZGlhc191Y2kiLCANCiAgICAgICAgICAgICAgICBteXRpdGxlID0gICJUb3RhbCBkw61hcyBVQ0kiLCANCiAgICAgICAgICAgICAgICBteWxhYmVsX3ggPSAiIiwgDQogICAgICAgICAgICAgICAgbXlsYWJlbF95ID0gIkRpYXMgVUNJIiwgDQogICAgICAgICAgICAgICAgbXlfZmlsbCA9ICIiKQ0KDQpwNiA8LSBteWJveHBsb3QobXlkYXRhID0gZmlsdGVyKGRhdGFfcmVob3NwLCBkaWFzX3VjZSA+IDApLCANCiAgICAgICAgICAgICAgICBteWV4cG9zdXJlID0gInJlaG9zcF9vbXMiLCANCiAgICAgICAgICAgICAgICBteW91dGNvbWUgPSAiZGlhc191Y2UiLCANCiAgICAgICAgICAgICAgICBteXRpdGxlID0gIlRvdGFsIGTDrWFzIFVDRSIsIA0KICAgICAgICAgICAgICAgIG15bGFiZWxfeCA9ICIiLCANCiAgICAgICAgICAgICAgICBteWxhYmVsX3kgPSAiRGlhcyBVQ0UiLCANCiAgICAgICAgICAgICAgICBteV9maWxsID0gIiIpDQoNCnAzIDwtIGdncGxvdGx5KHAzKQ0KcDQgPC0gZ2dwbG90bHkocDQpDQpwNSA8LSBnZ3Bsb3RseShwNSkNCnA2IDwtIGdncGxvdGx5KHA2KQ0KDQpzdWJwbG90KHA1LCBwNiwgcDMsIHA0LCANCiAgICAgICAgbnJvd3MgPSAyLCBuY29sKDIpLCANCiAgICAgICAgdGl0bGVYID0gVFJVRSwgdGl0bGVZID0gVFJVRSkgJT4lDQogIGxheW91dChzaG93bGVnZW5kID0gKEZBTFNFKSkNCg0KDQoNCmBgYA0KDQpTZSBwdWVkZSBvYnNlcnZhciBxdWUgZW4gbGEgdmFyaWFibGUgZMOtYXMgVUNJLCBjb3JyZXNwb25kaWVudGUgYWwgcHJpbWVyIGRpYWduw7NzdGljbywgbm8gcGFyZWNlIGhhYmVyIHVuYSBkaWZlcmVuY2lhIHNpZ25pZmljYXRpdmEgZW4gbGEgZGlzdHJpYnVjacOzbiBhbCBkaXNjcmltaW5hciBwb3IgbGEgdmFyaWFibGUgb2JqZXRpdm8gYmluYXJpYSwgZXMgZGVjaXIsIGVudHJlIGxvcyBjYXNvcyBkZSByZWhvc3BpdGFsaXphY2nDs24gKDEpIHkgY2Fzb3MgZGUgbm8gcmVob3NwaXRhbGl6YWNpw7NuICgwKS4gQWRpY2lvbmFsbWVudGUsIGxhIGRpc3RpYnVjacOzbiBlbiBhbWJhcyB2YXJpYWJsZXMgbm8gZXMgc2ltw6l0cmljYS4gDQoNCkEgcGVzYXIgZGUgcXVlIGxvcyBkYXRvcyBzZSBlbmN1ZW50cmFuIGJhc3RhbnRlIGRpc3BlcnNvcywgc2UgbG9ncmEgaWRlbnRpZmljYXIgZGlmZXJlbmNpYXMgZW4gbGEgdmFyaWFibGUgZGVsIHBhZ28gLWNvbiB2YWxvcmVzIG3DoXMgYWx0b3MgZW4gbG9zIGNhc28gZW4gcXVlIG5vIHRlcm1pbsOzIGRlIGJ1ZXZvIGhvc3BpdGFsaXphZG8sIHkgZW4gZWwgY2Fzw7MgZGVsIG7Dum1lcm8gZGUgZMOtYXMgaG9zcGl0YWxpemFkbyBsb3MgcmFuZ29zIHNvbiBtdWNobyBtw6FzIHBlcXVlw7FvcyBjdWFuZG8gaGF5IHVuYSByZWhvc3BpdGFsaXphY2nDs24uDQoNCjxocj4NCjxoMyBpZD0iQW5hbGlzaXNDYXIic3R5bGU9InRleHQtYWxpZ246IGNlbnRlcjsiIG1hcmtkb3duID0gIjEiPkFuw6FsaXNpcyB2YXJpYWJsZXMgY2F0ZWfDs3JpY2FzPC9oMz4NCjxkaXYgc3R5bGU9InRleHQtYWxpZ246IGp1c3RpZnkiPg0KDQpPYnNlcnZhbmRvIGxhcyB2YXJpYWJsZXMgY2F0ZWfDs3JpY2FzIGxhIGRpZmVyZW5jaWEgZW50cmUgbGEgcHJvYmFiaWxpZGFkIGRlIHF1ZSBlbCBldmVudG8gb2N1cnJhIChoYXlhIHJlaG9zcGl0YWxpemFjacOzbikgbyBubywgc2UgcHVlZGUgZXZpZGVuY2lhciBzw7NsbyBlbiBhbGd1bmFzIGNsYXNlcyBwb3IgY2F0ZWdvcsOtYSwgcGVybyBlbiBnZW5lcmFsLCBsYXMgcHJvcG9yY2lvbmVzIHN1ZWxlbiBzZXIgYmFzdGFudGVzIHNpbWlsYXJlcywgcG9yIGxvIHF1ZSBubyBlcyBwb3NpYmxlIGVsYWJvcmFyIGEgcHJpb3JpIHVuYSBoaXDDs3Rlc2lzIHF1ZSBlc3RpcHVsZSBkaWZlcmVuY2lhcyBzaWduaWZpY2F0aXZhcyBlbiBsYXMgZGlzdHJpYnVjaW9uZXMsIHBvciBsbyBtZW5vcyBwYXJhIG5pbmd1bmEgZGUgbGFzIGRvcyB2YXJpYWJsZXMgcmVsYWNpb25hZGFzIGVuIGVsIGdyw6FmaWNvIGEgY29udGludWFjacOzbi4NCg0KYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUV9DQoNCnA3IDwtIG15Z2VvbV9iYXIobXlkYXRhID0gZGF0YV9yZWhvc3AsIA0KICAgICAgICAgICAgICAgICBteWV4cG9zdXJlID0gImVkYWQiLCANCiAgICAgICAgICAgICAgICAgbXlvdXRjb21lID0gInJlaG9zcF9vbXMiLCANCiAgICAgICAgICAgICAgICAgbXl0aXRsZSA9ICJFZGFkIiwgDQogICAgICAgICAgICAgICAgIG15bGFiZWxfeCA9ICIiLCANCiAgICAgICAgICAgICAgICAgbXlsYWJlbF95ID0gIkZyZWN1ZW5jaWEiLCANCiAgICAgICAgICAgICAgICAgbXlfZmlsbCA9ICJSZWhvc3BpdGFsaXphY2nDs24iLCANCiAgICAgICAgICAgICAgICAgbXlfYW5nbGUgPSBOVUxMLA0KICAgICAgICAgICAgICAgICBteV9sZWdlbmQgPSAicmlnaHQiKQ0KDQpwOCA8LSBteWdlb21fYmFyKG15ZGF0YSA9IGRhdGFfcmVob3NwLCANCiAgICAgICAgICAgICAgICAgIG15ZXhwb3N1cmUgPSAiZXN0cmF0byIsIA0KICAgICAgICAgICAgICAgICAgbXlvdXRjb21lID0gInJlaG9zcF9vbXMiLCANCiAgICAgICAgICAgICAgICAgIG15dGl0bGUgPSAiRXN0cmF0byIsIA0KICAgICAgICAgICAgICAgICAgbXlsYWJlbF94ID0gIiIsIA0KICAgICAgICAgICAgICAgICAgbXlsYWJlbF95ID0gIkZyZWN1ZW5jaWEiLCANCiAgICAgICAgICAgICAgICAgIG15X2ZpbGwgPSAiUmVob3NwaXRhbGl6YWNpw7NuIiwgDQogICAgICAgICAgICAgICAgICBteV9hbmdsZSA9IE5VTEwsDQogICAgICAgICAgICAgICAgICBteV9sZWdlbmQgPSAibm9uZSIpDQoNCmdyaWQuYXJyYW5nZShwNywNCiAgICAgICAgICAgICBwOCkNCg0KYGBgDQoNClBvciBvdHJvIGxhZG8sIGVsIGF0cmlidXRvIHF1ZSBpbmRpY2EgZWwgaGVjaG8gZGUgcXVlIHNlIGhheWFuIHJlYWxpemFkbyBwcm9jZWRpbWllbnRvcyBxdWlyw7pyZ2ljb3MgZHVyYW50ZSBsYSBwcmltZXJhIGhvc3BpdGFsaXphY2nDs24gbXVlc3RyYW4gY2llcnRhIGRpZmVyZW5jaWEgZW4gbGEgZGlzdHJpYnVpw7NuIHBvciBncnVwbzsgZXMgbcOhcyBwcm9iYWJsZSBxdWUgbGEgcGVyc29uYSBkZWJhIHNlciByZWhvc3BpdGFsaXphZGEgZGUgbnVldm8uDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KDQpwOSA8LSBteWdlb21fYmFyKG15ZGF0YSA9IGRhdGFfcmVob3NwLCANCiAgICAgICAgICAgICAgICAgIG15ZXhwb3N1cmUgPSAicXVpcnVyIiwgDQogICAgICAgICAgICAgICAgICBteW91dGNvbWUgPSAicmVob3NwX29tcyIsIA0KICAgICAgICAgICAgICAgICAgbXl0aXRsZSA9ICJQcm9jIHF1aXLDunJnaWNvIiwgDQogICAgICAgICAgICAgICAgICBteWxhYmVsX3ggPSAiIiwgDQogICAgICAgICAgICAgICAgICBteWxhYmVsX3kgPSAiRnJlY3VlbmNpYSIsIA0KICAgICAgICAgICAgICAgICAgbXlfZmlsbCA9ICJSZWhvc3BpdGFsaXphY2nDs24iLCANCiAgICAgICAgICAgICAgICAgIG15X2FuZ2xlID0gTlVMTCwNCiAgICAgICAgICAgICAgICAgIG15X2xlZ2VuZCA9ICJyaWdodCIpDQoNCnAxMCA8LSBteWdlb21fYmFyKG15ZGF0YSA9IGRhdGFfcmVob3NwLCANCiAgICAgICAgICAgICAgICAgIG15ZXhwb3N1cmUgPSAiZXN0X2NpdmlsIiwgDQogICAgICAgICAgICAgICAgICBteW91dGNvbWUgPSAicmVob3NwX29tcyIsIA0KICAgICAgICAgICAgICAgICAgbXl0aXRsZSA9ICJFc3RhZG8gY2l2aWwiLCANCiAgICAgICAgICAgICAgICAgIG15bGFiZWxfeCA9ICIiLCANCiAgICAgICAgICAgICAgICAgIG15bGFiZWxfeSA9ICJGcmVjdWVuY2lhIiwgDQogICAgICAgICAgICAgICAgICBteV9maWxsID0gIlJlaG9zcGl0YWxpemFjacOzbiIsIA0KICAgICAgICAgICAgICAgICAgbXlfYW5nbGUgPSBOVUxMLA0KICAgICAgICAgICAgICAgICAgbXlfbGVnZW5kID0gInJpZ2h0IikNCg0KZ3JpZC5hcnJhbmdlKHA5LA0KICAgICAgICAgICAgICBwMTAsDQogICAgICAgICAgICAgIG5jb2wgPSAxLA0KICAgICAgICAgICAgICBucm93ID0gMikNCg0KYGBgDQoNCkNvbiBlbCBvYmpldGl2byBkZSBlbnJpcXVlY2VyIGVsIGFuw6FsaXNpcyBleHBsb3JhdG9yaW8sIHNlIGNhbGN1bGFyw6FuIGRvcyBtZWRpZGFzIG11eSBjb23Dum5lcyBkZSBsYSB0ZW9yw61hIGRlIGxhIGluZm9ybWFjacOzbiwgw6lzdGFzIHBlcm1pdGVuIGluZmVyaXIgYWxnbyBkZWwgcG9kZXIgcHJlZGljdGl2byBxdWUgcHVlZGVuIHRlbmVyIGxhcyB2YXJpYWJsZXMgaW5kZXBlbmRpZW50ZXMsIGFudGVzIGRlIGhhY2VyIHBhcnRlIGRlIHVuIG1vZGVsby4NCg0KW1ZvbHZlciBhbCDDjW5kaWNlXSgjaW5kaWNlKTwvZGl2Pg0KDQo8aHI+DQo8aDIgaWQ9IkFuYWxpc2lzV09FIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd24gPSIxIj5BbsOhbGlzaXMgZGUgY2xhc2lmaWNhY2nDs24gYmluYXJpYSB1c2FuZG8gV09FIHkgZWwgSVY8L2gyPg0KPGRpdiBzdHlsZT0idGV4dC1hbGlnbjoganVzdGlmeSI+DQoNCkVsIHBlc28gZGUgbGEgZXZpZGVuY2lhIChXT0UpIHkgZWwgdmFsb3IgZGUgbGEgaW5mb3JtYWNpw7NuIChJVikgYXl1ZGFuLCBlbnRyZSBvdHJhcyBjb3NhcywgYSBkZXRlcm1pbmFyIGxhIGNvbnRyaWJ1Y2nDs24gaW5kZXBlbmRpZW50ZSBkZSBjYWRhIHZhcmlhYmxlIGFsIHJlc3VsdGFkbywgeSBkZXRlY3RhciByZWxhY2lvbmVzIGxpbmVhbGVzIHkgbm8gbGluZWFsZXMuIEVsIFdPRSBtaWRlIGxhIHJlbGFjacOzbiBlbnRyZSBsYSB2YXJpYWJsZSBwcmVkaWN0aXZhIHkgZWwgb2JqZXRvIGJpbmFyaW8sIG1pZW50cmFzIHF1ZSBlbCBJViBtaWRlIGxhIGZ1ZXJ6YSBwcmVkaWN0aXZhIGRlIGVzYSByZWxhY2nDs24uDQoNCkxhIHRhYmxhIGEgY29udGludWFjacOzbiBjb250aWVuZSBsb3MgdmFsb3JlcyBkZWwgInZhbG9yIGRlIGxhIGluZm9ybWFjacOzbiIgY29uIHkgc2luIGVsIGFqdXN0ZSBkZXJpdmFkbyBkZSBsYSB2YWxpZGFjacOzbiBjcnV6YWRhLiBDdWFuZG8gc2UgcmVhbGl6YSBlbCBhanVzdGUgY29uIGVsIG9iamV0aXZvIGRlIHF1ZSBsb3MgcmVzdWx0YWRvcyBzZWFuIG3DoXMgZXN0YWJsZXMsIHRhbnRvIHBhZ28gZGVsIGRpYWduw7NzdGljbywgZWwgaGVjaG8gZGUgcXVlIGVsIHBhY2llbnRlIGhhbGxhIHBhc2FkbyBwb3IgbGEgVW5pZGFkIGRlIGN1aWRhZG9zLCB5IHNpIGZ1ZXJvbiByZWFsaXphZG9zIHByb2NlZGltaWVudG9zIHF1aXLDunJnaWNvcyBzZXLDoW4gbGFzIMO6bmljYXMgdmFyaWFibGVzIGNvbiBzdWZpY2llbnRlIGNhcGFjaWRhZCBkZSBwcmVkaWNjacOzbiBhIG5pdmVsIGluZGl2aWR1YWwgeSB1bml2YXJpYWJsZSAoSXYgPiAwLjA1KS4gQ3VhbmRvIHNlIHJlbGFqYSBlbCBzdXB1ZXN0bywgSVYgc2luIHJlc3RhciBlbCBwZW5hbHR5LCBzZSBpbmNsdWlyw61hbiBsb3MgZMOtYXMgZW4gcXVlIGVzdHV2byBob3NwaXRhbGl6YWRvLg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCnNldC5zZWVkKDEyMzQpDQpkYXRhX3JlaG9zcCA8LSBkYXRhX3JlaG9zcCAlPiUNCiAgbXV0YXRlKGlkID0gMTpucm93KC4pKSANCg0KZGF0YV9yZWhvc3AgJT4lDQogIHNhbXBsZV9mcmFjKHNpemUgPSAuNzApIC0+IHRyYWluDQoNCmRhdGFfcmVob3NwICU+JQ0KICBhbnRpX2pvaW4oeCA9IC4sDQogICAgICAgICAgICB5ID0gdHJhaW4sIA0KICAgICAgICAgICAgYnkgPSAiaWQiKSAtPiB0ZXN0DQogIA0KdHJhaW4gPC0gc2VsZWN0KC5kYXRhID0gdHJhaW4sIC1pZCkNCnRlc3QgPC0gc2VsZWN0KC5kYXRhID0gdGVzdCwgLWlkKQ0KDQpJViA8LSBjcmVhdGVfaW5mb3RhYmxlcyhkYXRhID0gdHJhaW4sDQogICAgICAgICAgICAgICAgICAgdmFsaWQgPSB0ZXN0LA0KICAgICAgICAgICAgICAgICAgIHkgPSAicmVob3NwX29tcyIpDQoNCmthYmxlX3N0eWxpbmcoa2FibGUoSVYkU3VtbWFyeSksIA0KICAgICAgICAgICAgICBwb3NpdGlvbiA9ICJjZW50ZXIiLCANCiAgICAgICAgICAgICAgcm93X2xhYmVsX3Bvc2l0aW9uID0gMSwNCiAgICAgICAgICAgICAgZnVsbF93aWR0aCA9IEYpDQpgYGANCg0KRGUgYWN1ZXJkbyBhbCBwb2RlciBwcmVkaWN0aXZvIGRlIGNhZGEgdW5hIGRlIGxhcyB2YXJpYWJsZXMsIHNlIGVsaWdlbiBhcXVlbGxhcyBjdXlvIFZhbG9yIGRlIGxhIGluZm9ybWFjacOTbiAoSVYpIHNlYSBzdXBlcmlvciBhbCAyJSAoMCwwMikuIExhcyB2YXJpYWJsZXMgY29uIElWIGluZmVyaW9yZXMgYSBlc3RlIHZhbG9yIHNlIGNvbnNpZGVyYW4gaW1wcmVkaWN0aXZhcyB5IHNlIGRlY2lkZSBkZXNjYXJ0YXJsYXMuIExhcyB2YXJpYWJsZXMgcXVlIGNvbnRpbnVhbiwgZW4gb3JkZW4gZGUgcmVsZXZhbmNpYSBzZWd1biBzdSBwb2RlciBwcmVkaWN0b3IsIHNvbjoNCg0KPHVsPg0KPGxpPnBhZ29faG9zcDwvbGk+DQo8bGk+cXVpcnVyPC9saT4NCjxsaT5kaWFzX3VjZTwvbGk+DQo8bGk+ZGlhc191Y2k8L2xpPg0KPGxpPnByb3ZlZWRvcjwvbGk+DQo8bGk+ZGlhc19ob3NwPC9saT4NCjxsaT5lZGFkPC9saT4NCjxsaT5nZW5lcm88L2xpPg0KPGxpPmNpdWRhZDwvbGk+DQo8L3VsPg0KDQoNClNpbiBlbWJhcmdvLCB0YW50byBsYSBjaXVkYWQsIGNvbW8gZWwgcHJvdmVlZG9yIG5vIHNlcsOhbiB0ZW5pZG9zIGVuIGN1ZW50YSwgcG9yIHF1ZSBwdWVkZW4gbGxlZ2FyIGEgY29uZGljaW9uYXIgbnVlc3RyYSB2YXJpYWJsZSBvYmpldGl2by4gQWRpY29uYWxtZW50ZSwgbG8gZMOtYXMgVUNFIHkgVUNJIHBhcmVjZW4gZXN0YXIgYWx0YW1lbnRlIGNvcnJlbGFjaW9uYWRvcyBjb24gbGEgdmFyaWFibGUgb2JqZXRpdm8uIHBvciBsbyBxdWUgdGFtcG9jbyBzZXLDoW4gdGVuaWRvcyBlbiBjdWVudGEuDQoNCkVuZm9jYW5kb25vcyBlbiBlbCBwYWdvIGRlbCBkaWFnbsOzc3RpY28sIGVsIGN1YWwsIGVzIHVuYSBkZSBsYXMgdmFyaWFibGVzIGNvbiBtYXlvciBpbmZsdWVuY2lhLCBlbCBXT0Ugbm9zIGluZGljYSB1bmEgcmVsYWNpw7NuIG5vIGxpbmVhbCwgY29uIHVuIGluY3JlbWVudG8gZW4gZWwgV09FIGEgbWVkaWRhIHF1ZSBkaXNtaW51eWUgZWwgcmFuZ28gZGUgcGFnbyBlbiBlbCBkaWFnbsOzc3RpY28uDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0Ka2FibGVfc3R5bGluZyhrYWJsZShJViRUYWJsZXMkcGFnb19ob3NwKSwgDQogICAgICAgICAgICAgIHBvc2l0aW9uID0gImNlbnRlciIsIA0KICAgICAgICAgICAgICByb3dfbGFiZWxfcG9zaXRpb24gPSAxLA0KICAgICAgICAgICAgICBmdWxsX3dpZHRoID0gRikNCmBgYA0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCm4gPC0gbmFtZXMoSVYkVGFibGVzKQ0KZm9yIChpIGluIDE6bGVuZ3RoKG4pKXsNCiAgIHBsb3RfaW5mb3RhYmxlcyhJViwgbltpXSl9DQoNCk11bHRpUGxvdChJViwgSVYkU3VtbWFyeSRWYXJpYWJsZVtjKDEsMiwzLDQsNiwxMCldKQ0KYGBgDQoNCkNvbW8gc2UgcHVkbyBvYnNlcnZhciBlbiBlbCBhbmFsaXNpcyBkZWwgV09FLCBlc3RhIHTDqWNuaWNhIGFqdXN0YSBsb3MgdmFsb3JlcyBkZSBsYXMgdmFyaWFibGVzIG51bWVyaWNhcyBlbiByYW5nb3MgYWNvdGFkb3MgZGUgYWN1ZXJkbyBhbCB2YWxvciBkZSBsYSBpbmZvcm1hY2lvbiBkZSBjYWRhIHVuYSBkZSBlbGxhcyBlbiByZWxhY2lvbiBjb24gbGEgdmFyaWFibGUgZGVwZW5kaWVudGUuIFBvciBlc3RvLCBlcyBpbXBvcnRhbnRlIHRyYW5zZm9ybWFyIGRpY2hhcyB2YXJpYWJsZXMgZW4gbG9zIHJhbmdvcyByZWNvbWVuZGFkb3MuDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KICMgZGF0YV9yZWhvc3AgJT4lDQogIyAgIG11dGF0ZShwYWdvX2hvc3AgPSBjYXNlX3doZW4oIC4kcGFnb19ob3NwIDw9IDEwNzgyMCB+ICJbMCwxMDc4MjBdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDEwODIwMCAmIC4kcGFnb19ob3NwIDw9IDc1OTgyOCB+ICJbMTA4MjAwLDc1OTgyOF0iLA0KICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgLiRwYWdvX2hvc3AgPj0gNzYwMDk1ICYgLiRwYWdvX2hvc3AgPD0gMTUzMTQwOCB+ICJbNzYwMDk1LDE1MzE0MDhdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDE1MzE2MTggJiAuJHBhZ29faG9zcCA8PSAyMTMxNTkyIH4gIlsxNTMxNjE4LDIxMzE1OTJdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDIxMzIxMTUgJiAuJHBhZ29faG9zcCA8PSAyODk1MDA0IH4gIlsyMTMyMTE1LDI4OTUwMDRdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDI4OTUwMDUgJiAuJHBhZ29faG9zcCA8PSAzODcwODAzIH4gIlsyODk1MDgzLDM4NzA4MDNdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDM4NzA4NTMgJiAuJHBhZ29faG9zcCA8PSA0OTQ0MTIzIH4gIlszODcwODUzLDQ5NDQxMjNdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDQ5NDQxMjQgJiAuJHBhZ29faG9zcCA8PSA2NjUyNzI2IH4gIls0OTQ0NDM2LDY2NTI3MjZdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kcGFnb19ob3NwID49IDY2NTMxODEgJiAuJHBhZ29faG9zcCA8PSAxMDI2NjkzNCB+ICJbNjY1MzE4MSwxMDI2NjkzNF0iLA0KICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgLiRwYWdvX2hvc3AgPj0gMTAyNzIyNjggJiAuJHBhZ29faG9zcCA8PSAzNTUxNDc1MSB+ICJbMTAyNzIyNjgsMzU1MTQ3NTFdIiksDQogIyAgICAgICAgICBkaWFzX2hvc3AgPSBjYXNlX3doZW4oIC4kZGlhc19ob3NwID09IDEgfiAiWzFdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLiRkaWFzX2hvc3AgPT0gMiB+ICJbMl0iLA0KICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAuJGRpYXNfaG9zcCA9PSAzIH4gIlszXSIsDQogIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4kZGlhc19ob3NwID09IDQgfiAiWzRdIiwNCiAjICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLiRkaWFzX2hvc3AgPj0gNSAmIC4kZGlhc19ob3NwIDw9IDcgfiAiWzUsN10iLA0KICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAuJGRpYXNfaG9zcCA+PSA4ICYgLiRkaWFzX2hvc3AgPD0gMzAgfiAiWzgsMzBdIiksDQogIyAgICAgICAgICBwYWdvX2hvc3AgPSBhcy5mYWN0b3IocGFnb19ob3NwKSwNCiAjICAgICAgICAgIGRpYXNfaG9zcCA9IGFzLmZhY3RvcihkaWFzX2hvc3ApKSAtPiBkYXRhX3JlaG9zcA0KICMgDQogIyBzdHIoZGF0YV9yZWhvc3ApDQogDQogDQpkYXRhX3JlaG9zcCA8LSBkYXRhX3JlaG9zcCAlPiUgbXV0YXRlX2VhY2hfKGZ1bnMoc2NhbGUoLikgJT4lIGFzLnZlY3RvciksIA0KICAgICAgICAgIHZhcnM9YygicGFnb19ob3NwIiwiZGlhc19ob3NwIikpIA0KDQpgYGANCg0KDQpbVm9sdmVyIGFsIMONbmRpY2VdKCNpbmRpY2UpPC9kaXY+DQoNCjxocj4NCjxoMiBpZD0iTW9kZWxvIiBzdHlsZT0idGV4dC1hbGlnbjogY2VudGVyOyIgbWFya2Rvd249IjEiPk1vZGVsbzwvYT48L2gyPg0KPGRpdiBzdHlsZSA9ICJ0ZXh0LWFsaWduOiBqdXN0aWZ5Ij4NCg0KRWwgb2JqZXRpdm8gcHJpbmNpcGFsIGRlbCBhbsOhbGlzaXMgZXMgZXN0aW1hciB1biBtb2RlbG8gcHJlZGljdGl2byBjb24gZWwgY3XDoWwgc2UgcHVlZGEgZXN0aW1hciBsYSBwcm9iYWJpbGlkYWQgZGUgcXVlIHVuIHBhY2llbnRlIHRlcm1pbmUgZW4gdW5hIHJlaG9zcGl0YWxpemFjacOzbiwgYXNvY2lhZGEgYSB1biBkaWFuZ8Ozc3RpY28gYW50ZXJpb3IuIFBhcmEgZWxsbyBzZSBlbXBsZWFyw6EgdW4gbW9kZWxvIGRlIHJlZ3Jlc2nDs24gbG9nw61zdGljYSwgZWwgY3XDoWwgZXMgYW1wbGlhbWVudGUgdXRpbGl6YWRvIHBhcmEgcmVzb2x2ZXIgcHJvYmxlbWFzIGRlIGNsYXNpZmljYWNpw7NuIGJpbmFyaWEuDQoNClVuYSB2ZXogc2UgcmVhbGl6YW4gbG9zIGZpbHRyb3MgZGUgY2FsaWRhZCB5IGNvbXBsZXRpdHVkLCB5IHRyYXMgbG8gb2J0ZW5pZG8gZW4gbG9zIHJlc3VsdGFkb3MgZGVsIFdPRSwgc2UgcHJvY2VkZSBhIHJlYWxpemFyIGxhIHNlbGVjY2lvbiBkZSB2YXJpYWJsZXMgcGFyYSBlbCBtb2RlbG8uIFNlIHRlbmRyw6FuIGVuIGN1ZW50YSBlbnRvbmNlcywgZWwgcGFnbyByZWFsaXphZG8sIGxvcyBkw61hcyBlbiBxdWUgZXN0dXZvIGVsIHBhY2llbnRlIGRlIGZvcm1hIGdlbmVyYWwsIGVsIGhlY2hvIGRlIHF1ZSBzZSBsZSBoYXlhIHJlYWxpemFkbyBvIG5vIHVuYSBjaXJ1Z8OtYSwgZWwgZ8OpbmVybywgbGEgZWRhZCB5IGVsIGVzdHJhdG8uDQoNClBhcmEgZXZhbHVhciBsYSBjYXBhY2lkYWQgZGUgZ2VuZXJhbGl6YWNpw7NuIGRlbCBtb2RlbG8sIHNlIGRpdmlkaXLDoSBlbCBjb25qdW50byBkZSBkYXRvcyBlbiBlbnRyZW5hbWllbnRvICg3MCUpIHkgcHJ1ZWJhICgzMCUpLg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCnNldC5zZWVkKDEyMzQpDQpkYXRhX3JlaG9zcCAlPiUNCiAgc2VsZWN0KGlkLA0KICAgICAgICBwYWdvX2hvc3AsDQogICAgICAgIHF1aXJ1ciwNCiAgICAgICAgZGlhc19ob3NwLA0KICAgICAgICBnZW5lcm8sDQogICAgICAgIGVkYWQsDQogICAgICAgIGVzdHJhdG8sDQogICAgICAgIHJlaG9zcF9vbXMpIC0+IG1vZGVsX3JlaG9zcA0KDQptb2RlbF9yZWhvc3AgJT4lDQogIHNhbXBsZV9mcmFjKHNpemUgPSAwLjcpIC0+IHRyYWluaW5nDQoNCm1vZGVsX3JlaG9zcCAlPiUNCiAgYW50aV9qb2luKHggPSAuLA0KICAgICAgICAgICAgeSA9IHRyYWluaW5nLA0KICAgICAgICAgICAgYnkgPSAiaWQiKSAtPiB0ZXN0aW5nDQoNCnRlc3RpbmcgJT4lDQogIHNlbGVjdCgtaWQpIC0+IHRlc3RpbmcNCg0KdHJhaW5pbmcgJT4lDQogIHNlbGVjdCgtaWQpICU+JQ0KICBtdXRhdGUocmVob3NwX29tcyA9IGFzLmZhY3RvcihyZWhvc3Bfb21zKSkgLT4gdHJhaW5pbmcNCg0KYGBgDQoNCltWb2x2ZXIgYWwgw41uZGljZV0oI2luZGljZSk8L2Rpdj4NCg0KPGhyPg0KPGgzIGlkPSJTTU9URSIgc3R5bGU9InRleHQtYWxpZ246IGNlbnRlcjsiIG1hcmtkb3duPSIxIj5TbW90ZTwvaDM+DQo8ZGl2IHN0eWxlPSJ0ZXh0LWFsaWduOiBqdXN0aWZ5Ij4NCg0KQ29tbyBzZSBoYWJpYSBtZW5jaW9uYWRvIGFudGVyaW9ybWVudGUsIGxhIGluZm9ybWFjaW9uIHNlIGVuY3VlbnRyYSBkZXNiYWxhbmNlYWRhOyBlc3RvIGVzLCB0ZW5pZW5kbyBlbiBjdWVudGEgcXVlIGVsIHByb2JsZW1hIGVuIHF1ZSBzZSBlc3RhIHRyYWJhamFuZG8gY29uc2lzdGUgZW4gbGEgY2xhc2lmaWNhY2lvbiBkZSB1bmEgdmFyaWFibGUgZGljb3TDs21pY2EsIHNlIGRlYmUgYW5hbGl6YXIgZWwgbml2ZWwgZGUgcmVwcmVzZW50YWNpb24gZGUgc3VzIHBvc2libGVzIHZhbG9yZXMgZGVudHJvIGRlbCBjb25qdW50byB0b3RhbCBkZSBkYXRvcy4NCg0KYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUV9DQoNCmthYmxlKGFzLmRhdGEuZnJhbWUocHJvcC50YWJsZSh0YWJsZShtb2RlbF9yZWhvc3AkcmVob3NwX29tcykpKSkgJT4lDQprYWJsZV9zdHlsaW5nKHBvc2l0aW9uID0gImNlbnRlciIsIA0KICAgICAgICAgICAgICByb3dfbGFiZWxfcG9zaXRpb24gPSAxLA0KICAgICAgICAgICAgICBmdWxsX3dpZHRoID0gRikgJT4lDQpyb3dfc3BlYygwLGJhY2tncm91bmQ9IiNFQkYwRjciKQ0KDQoNCmBgYA0KDQpWZW1vcyBxdWUgbGEgcmVwcmVzZW50YWNpb24gcGFyYSBsYSBjYXRlZ29yw61hIHBvc2l0aXZhIGVzIHVuIHBvY28gbWFzIGRlbCAyJSBkZSBsYSBpbmZvcm1hY2nDs24uIEVuIGVzdGUgY2FzbyB2YW1vcyBhIHJlYWxpemFyIHVuIHRyYXRhbWllbnRvIHF1ZSBwZXJtaXRhIGF1bWVudGFyIGxhIGNsYXNlIG1pbm9yaXRhcmlhLCBzaW4gdXRpbGl6YXIgc29sdWNpb25lcyBnZW7DqXJpY2FzIGNvbW8gcmVkdWNpciBsYSBjbGFzZSBtYXlvcml0YXJpYSBhbCBuaXZlbCBkZSBsYSBjbGFzZSBtZW5vci4NCg0KUGFyYSBlbGxvLCB2YW1vcyBhIHV0aWxpemFyIGxhIHTDqWNuaWNhIFNNT1RFIChTeW50aGV0aWMgTWlub3JpdHkgT3ZlcnNhbXBsaW5nIE1ldGhvZCksIGxhIGN1YWwgZ2VuZXJhIG51ZXZhcyBpbnN0YW5jaWFzIGFydGlmaWNpYWxlcyBkZSBsYSBjbGFzZSBtw6FzIHBlcXVlw7FhIGludGVycG9sYW5kbyBsb3MgdmFsb3JlcyBkZSBsYXMgaW5zdGFuY2lhcyBtaW5vcml0YXJpYXMgbcOhcyBjZXJjYW5hcyBhIHVuYSBkYWRhLg0KDQpQb3IgbWVkaW8gZGUgU01PVEUgc2UgZ2VuZXJhcsOhIHVuIG51ZXZvIHNldCBkZSBkYXRvcyBkZSBlbnRyZW5hbWllbnRvLCBlbiBlbCBjdWFsIHNlIHRlbmdhIHVuIDYwJSBkZSBpbmZvcm1hY2lvbiBwYXJhIGxhIGNhdGVnb3JpYSBuZWdhdGl2YSAocmVob3NwX29tcyA9IDApIHkgNDAlIHBhcmEgbGEgY2F0ZWdvcmlhIHBvc2l0aXZhIChyZWhvc3Bfb21zID0gMCkuDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZWNobyA9IEZBTFNFLCBldmFsID1UUlVFfQ0KdHJhaW5pbmcgPC0gU01PVEUocmVob3NwX29tcyB+IC4sIGFzLmRhdGEuZnJhbWUodHJhaW5pbmcpLCBwZXJjLm92ZXIgPSAzNTAsIHBlcmMudW5kZXIgPSAxNTApDQojdHJhaW5pbmcgPC0gU01PVEUocmVob3NwX29tcyB+IC4sIGFzLmRhdGEuZnJhbWUodHJhaW5pbmcpLCBwZXJjLm92ZXIgPSAzMDAsIHBlcmMudW5kZXIgPSAyMDApDQpgYGANCg0KVmVyaWZpY2Ftb3MgcXVlIGVsIHNldCBkZSBlbnRyZW5hbWllbnRvIHNlIGVuY3VlbnRyZSBiYWxhbmNlYWRvOg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCg0Ka2FibGVfc3R5bGluZyhrYWJsZShhcy5kYXRhLmZyYW1lKHByb3AudGFibGUodGFibGUodHJhaW5pbmckcmVob3NwX29tcykpKSksIA0KICAgICAgICAgICAgICBwb3NpdGlvbiA9ICJjZW50ZXIiLCANCiAgICAgICAgICAgICAgcm93X2xhYmVsX3Bvc2l0aW9uID0gMSwNCiAgICAgICAgICAgICAgZnVsbF93aWR0aCA9IEYpICU+JQ0Kcm93X3NwZWMoMCxiYWNrZ3JvdW5kPSIjRUJGMEY3IikNCg0KYGBgDQoNCltWb2x2ZXIgYWwgw41uZGljZV0oI2luZGljZSk8L2Rpdj4NCg0KPGhyPg0KPGgzIGlkPSJBanVzdGVtb2QiIHN0eWxlPSJ0ZXh0LWFsaWduOiBjZW50ZXI7IiBtYXJrZG93bj0iMSI+QWp1c3RlIGRlbCBtb2RlbG8geSBFc3RpbWFjacOzbiBkZSBwYXLDoW1ldHJvczwvaDM+DQo8ZGl2IHN0eWxlPSJ0ZXh0LWFsaWduOiBqdXN0aWZ5Ij4NCg0KRGVsIHJlc3VsdGFkbyBleHBsb3JhdG9yaW8gYW50ZXJpb3IsIGFsIGRpc2NyaW1pbmFyIGVsIGFuw6FsaXNpcyBkZSBsYXMgdmFyaWFibGVzIGluZGVwZW5kaWVudGVzIHBvciBudWVzdHJhIHZhcmlhYmxlIG9iamV0aXZvIChSZWhvc3BpdGFsaXphY2nDs24pLCBlcyBwb3NpYmxlIGV2aWRlbmNpYXIgdW5hIGRpZmVyZW5jaWEgY2xhcmEgZW50cmUgbGFzIGRpc3RyaWJ1Y2lvbmVzIHBhcmEgbG9zIGF0cmlidXRvczogUGFnby9jb3N0byBkZWwgcHJvY2VkaW1pZW50byB5IGxvcyBkw61hcyBlbiBxdWUgZWwgdXN1YXJpbyBlc3R1dm8gaW50ZXJuYWRvIHlhIHNlYSBlbiBsYSBVbmlkYWQgZGUgQ3VpZGFkb3MgSW50ZW5zaXZvcyBvIEVzcGVjaWFsZXMuIEVzdG8gcG9kcsOtYSBzZXIgdW4gaW5kaWNpbyBkZSBxdWUgZXN0YXMgdmFyaWFibGVzIGVuIHBhcnRpY3VsYXIsIHB1ZWRlbiBsbGVnYXIgYSBzZXIgcmVsZXZhbnRlcyBwYXJhIGV4cGxpY2FyIGxhIHByb2JhYmlsaWRhZCBkZSBvY3VycmVuY2lhIGRlbCBldmVudG8sIGVzIGRlY2lyLCBjdWFuZG8gaHVibyB1bmEgaG9zcGl0YWxpemFjacOzbiBwb3N0ZXJpb3IgbGlnYWRhIGEgdW4gZGlhZ27Ds3N0aWNvLg0KDQpBIGNvbnRpbnVhY2nDs24sIGFsIGFqdXN0YXIgZWwgbW9kZWxvIG9idGVuZW1vcyBsb3Mgc2lndWllbnRlcyByZXN1bHRhZG9zOg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUV9DQpybShsaXN0PWxzKClbIWxzKCkgJWluJSBjKCJ0cmFpbmluZyIsICJ0ZXN0aW5nIiwgImRhdGFfcmVob3NwIildKQ0KbXlsb2dpdCA8LSBnbG0ocmVob3NwX29tcyB+IHBhZ29faG9zcCArIHF1aXJ1ciArIGRpYXNfaG9zcCArIGVkYWQgKyBnZW5lcm8gKyBlc3RyYXRvLCBkYXRhID0gdHJhaW5pbmcsIGZhbWlseSA9ICJiaW5vbWlhbCIpDQpzdW1tYXJ5KG15bG9naXQpDQpgYGANCg0KMS4gQ2FkYSBjYW1iaW8gZW4gdW5hIHVuaWRhZCBlbiBlbCBwYWdvIGhvc3BpdGFsYXJpbyBkaXNtaW51aXLDoSBsYXMgcHJvYmFiaWxpZGFkZXMgZGUgcmVob3NwaXRhbGl6YWNpw7NuLCBwZXJvIGVuIHVuYSBjYW50aWRhZCBtdXkgcGVxdWXDsWEgKC02Ljk1M0UtMDgpDQoyLiBDdWFuZG8gYSB1biBwYWNpZW50ZSBzZSBsZSByZWFsaXphIHVuIHByb2NlZGltaWVudG8gcXVpcsO6cmdpY28gc3UgcHJvYmFiaWxpZGFkIGRlIHF1ZSB0ZXJtaW5lIGhvc3BpdGFsaXphZG8gZGUgbnVldm8gcG9yIGVsIG1pc21vIGRpYWduw7NzdGljbywgZGlzbWludXllIGVuIG3DoXMgZGVsIDE5JSBlbiBjb21wYXJhY2nDs24gYSBjdWFuZG8gbm8gc2UgbGUgcmVhbGl6YSBuaW5ndW5hIGNpcnVnw61hLg0KMy4gTGEgcHJvYmFiaWxpZGFkIGRlIHJlaG9zcGl0YWxpemFjacOzbiBkZWNyZWNlLCBhbCBwZXJtYW5lY2VyIHVuIGTDrWEgYWRpY2lvbmFsIGVsIHBhY2llbnRlIGVuIGxhIFVDRSAoZW4gMS40NSkgeSBsYSBVQ0kgKDEuNzMpDQoNCkVsIHJlc3RvIGRlIGxhcyB2YXJpYWJsZXMgbm8gc29uIHN1ZmljaWVudGVtZW50ZSBleHBsaWNhdGl2YXMgcGFyYSBwcmVkZWNpciwgZGUgbWFuZXJhIHNpZ25pZmljYXRpdmEsIHN1IGVmZWN0byBzb2JyZSBsYSB2YXJpYWJsZSBkZSByZXNwdWVzdGEgYmluYXJpYS4NCg0KRGVzcHXDqXMgZGUgZXN0aW1hZG9zIGxvcyBjb2VmaWNlbnRlcyBzZSBwcm9jZWRlIGEgcmVhbGl6YXIgbGEgcHJlZGljY2nDs24gZGVudHJvIHkgZnVlcmEgZGUgbXVlc3RyYSBwYXJhIGV2YWx1YXIgbGEgcHJlY2lzacOzbiAoYWNjdXJhY3kpIHkgY2FwYWNpZGFkIGRlIGdlbmVyYWxpemFjacOzbiBkZSBudWVzdHJvIG1vZGVsby4gDQoNCmBgYHtyLCBtZXNzYWdlID0gRkFMU0UsIGVjaG8gPSBGQUxTRSwgZXZhbCA9VFJVRX0NCnByZWRfdHJhaW4gPC0gcHJlZGljdChteWxvZ2l0LCBuZXdkYXRhID0gdHJhaW5pbmdbLTddLCB0eXBlID0gInJlc3BvbnNlIikNCnlfcHJlZF90cmFpbiA8LSBpZmVsc2UocHJlZF90cmFpbiA+IDAuNSwgMSwgMCkNCnlfYWN0X3RyYWluIDwtIHRyYWluaW5nJHJlaG9zcF9vbXMNCg0KcHJlZCA9IHByZWRpY3QobXlsb2dpdCwgdHlwZSA9ICdyZXNwb25zZScsIG5ld2RhdGEgPSB0ZXN0aW5nWy03XSkNCnlfcHJlZCA9IGlmZWxzZShwcmVkID4gMC41LCAxLCAwKQ0KeV9hY3QgPC0gdGVzdGluZyRyZWhvc3Bfb21zDQoNCnN0cih0cmFpbmluZykNCg0KDQprYWJsZShkYXRhLmZyYW1lKFRyYWluID0gbWVhbih5X3ByZWRfdHJhaW4gPT0geV9hY3RfdHJhaW4pLCBUZXN0ID0gbWVhbih5X3ByZWQgPT0geV9hY3QpKSkgJT4lDQogIGthYmxlX3N0eWxpbmcocG9zaXRpb24gPSAiY2VudGVyIiwgDQogICAgICAgICAgICAgICAgcm93X2xhYmVsX3Bvc2l0aW9uID0gMSwNCiAgICAgICAgICAgICAgICBmdWxsX3dpZHRoID0gRikgJT4lDQpyb3dfc3BlYygwLGJhY2tncm91bmQ9IiNFQkYwRjciKQ0KICANCmBgYA0KDQpMb3MgcmVzdWx0YWRvcyBpbmRpY2FuIHVuIG5pdmVsIGRlIGFjY3VyYWN5IHNvc3BlY2hvc2FtZW50ZSBlbGV2YWRvLCBpbmNsdXNvIGVsIG1vZGVsbyBwYXJlY2UgYWp1c3RhcnNlIG1lam9yIGNvbiBsb3MgZGF0b3MgbnVldm9zLCBxdWUgZW4gbGEgcGFydGUgZGUgZW50cmVuYW1pZW50by4gUGFyYSB2ZXIgZW4gZGV0YWxsZSBjb21vIHNlIGNvbXBvcnRhLCBhbCBkaXNjcmltaW5hciBlbnRyZSBsb3MgY2Fzb3MgZW4gcXVlIGVsIHBhY2llbnRlIHNhbGUgZGVmaW5pdGl2YW1lbnRlIG8gdGVybWluYSBlbiB1bmEgcmVob3NwaXRhbGl6YWNpw7NuLCB5IGV2aWRlbmNpYXIgc3UgZGVzZW1wZcOxbyBwb3Igc2VwYXJhZG8sIHNlIGVzdGltYXLDoSBsYSBtYXRyaXogZGUgY29uZnVzacOzbjoNCg0KYGBge3J9DQp0YWJsZShhcy5tYXRyaXgodGVzdGluZ1ssIDddKSwgeV9wcmVkID4gMC41KQ0KYGBgDQoNCkxvcyByZXN1bHRhZG9zIG5vIHBhcmVjZW4gaW5kaWNhciBxdWUgw6lzdGUgY29tcG9ydGFtaWVudG8gc2UgZGVuIGEgY2F1c2EgZGVsIGRlc2JhbGFuY2VvLiBQb3IgdW4gbGFkbyB0ZW5lbW9zIHF1ZSBsYSBzZW5zaWJpbGlkYWQgeSBsYSBlc3BlY2lmaWNpZGFkIGNvcnJlc3BvbmRlbiBhbCA5NiUgeSA5OSUuIFNpIHNlIGhhYmxhIGVuIHTDqXJtaW5vcyBkZSBwcmVjaXNpw7NuLCBzZXLDrWEgZGUgZGUgdW4gOTklIGN1YW5kbyBlcyAwLCBlcyBkZWNpciBkZSB1biB0b3RhbCBkZSAxMDIxOCBwYWNpZW50ZXMgcXVlIG5vIHRlcm1pbmFyb24gZW4gaG9zcGl0YWxpemFjacOzbiBwdWRlIHByZWRlY2lyIGNvbiB1bmEgZXhhY3RpdHVkIHF1ZSBlbCA5OSUgbm8gbG8gaGFyw61hbiwgbWllbnRyYXMgcXVlIGRlIDI0OSBwYWNpZW50ZXMgcXVlIHNpIHR1dmllcm9uIHVuYSByZWhvc3BpdGFsaXphY2nDs24gcHVkZSBwcmVkZWNpciBxdWUgMjEwIGVmZWN0aXZhbWVudGUgbG8gaGFyw61hbiwgZXMgZGVjaXIsIG1pIHByZWNpc2nDs24gZnVlIGRlbCA4NCUuIA0KDQpTaW4gZW1iYXJnbywgbm8gc8OzbG8gcG9yIGxvcyByZXN1bHRhZG9zIG9idGVuaWRvcyBjb24gZWwgYWp1c3RlIGRlbCBtb2RlbG8sIHNpbm8gdGFtYmnDqW4gY29uIGxvIHF1ZSB2ZcOtYW1vcyBhbnRlcmlvcm1lbnRlIGVuIGxhIHRhYmxhIGRlbCB2YWxvciBkZSBsYSBpbmZvcm1hY2nDs24sIGxhcyB2YXJpYWJsZXMgZMOtYXMgVUNJIHkgZMOtYXMgVUNFIHBhcmVjZW4gZXhwbGljYXIgZGUgbWFuZXJhIGNhc2kgcGVyZmVjdGEgbGEgcHJvYmFiaWxpZGFkIGRlIHF1ZSBvY3VycmUgdW4gZXZlbnRvIHJlaG9zcGl0YWxhcmlvIHkgZXN0byBwdWVkZSBkZWJlcnNlIC4uLi4uLi4uIHBvciBlbmRlIHNlIGRlY2lkZSBlc3RpbWFyIGVsIG1vZGVsbyBzaW4gaW5jbHVpcmxhcy4NCg0KQSBjb250aW51YWNpw7NuLCBzZSBvYnRpZW5lbiBsb3MgcmVzdWx0YWRvczoNCg0KDQpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCBlY2hvID0gRkFMU0UsIGV2YWwgPVRSVUV9DQpsaWJyYXJ5KFJPQ1IpDQpST0NScHJlZCA9IHByZWRpY3Rpb24ocHJlZCwgdGVzdGluZyRyZWhvc3Bfb21zKQ0KIA0KIyBQZXJmb3JtYW5jZSBmdW5jdGlvbg0KUk9DUnBlcmYgPSBwZXJmb3JtYW5jZShST0NScHJlZCwgInRwciIsICJmcHIiKQ0KDQpwZXJmMSA8LSBwZXJmb3JtYW5jZShST0NScHJlZCwgInByZWMiLCAicmVjIikNCnBsb3QocGVyZjEpDQogDQojIFBsb3QgUk9DIGN1cnZlDQpwbG90KFJPQ1JwZXJmLCBjb2xvcml6ZT1UUlVFLCBwcmludC5jdXRvZmZzLmF0PXNlcSgwLDEsYnk9MC4xKSwgdGV4dC5hZGo9YygtMC4yLDEuNykpDQoNCmBgYA0KDQojIyNSZWd1bGFyaXphZG8NCg0KcGFyYSBsYSByZWd1bGFyaXphY2nDs24gc2UgdXRpbGl6YXLDoSBlbCBwYXF1ZXRlIGdsbW5ldCwgZWwgY3VhbCBhc2lnbmEgdW4gdmFsb3IgZGUgYWxwaGEgPSAwIGkgZXMgcmlkZ2UgeSBhbHBoYSA9IDEgc2kgbGFzc28uIEFudGVzIGRlIHN1bWVyZ2lyc2UgZW4gZWwgY8OzZGlnbywgdmFsZSBsYSBwZW5hIHNlw7FhbGFyIGxvIHNpZ3VpZW50ZToNCg0KKiBnbG1uZXQgbm8gdGllbmUgdW5hIGludGVyZmF6IGRlIGbDs3JtdWxhLCBwb3IgbG8gcXVlIHVubyB0aWVuZSBxdWUgaW5ncmVzYXIgbG9zIHByZWRpY3RvcmVzIGNvbW8gdW5hIG1hdHJpeiB5IGxhcyBldGlxdWV0YXMgZGUgY2xhc2UgY29tbyB1biB2ZWN0b3IuDQoNCiogbm8gYWNlcHRhIHByZWRpY3RvcmVzIGNhdGVnw7NyaWNvcywgcG9yIGxvIHF1ZSB1bm8gdGllbmUgcXVlIGNvbnZlcnRpcmxvcyBlbiB2YWxvcmVzIG51bcOpcmljb3MgYW50ZXMgZGUgcGFzYXJsb3MgYSBnbG1uZXQuDQoNCkxhIGZ1bmNpw7NuIGdsbW5ldCBtb2RlbC5tYXRyaXggY3JlYSBsYSBtYXRyaXogeSB0YW1iacOpbiBjb252aWVydGUgbG9zIHByZWRpY3RvcmVzIGNhdGVnw7NyaWNvcyBlbiB2YXJpYWJsZXMgZmljdGljaWFzIGFwcm9waWFkYXM7IG1pZW50cmFzIHF1ZSBzZSB1c2Fyw6EgbGEgZnVuY2nDs24gY3YuZ2xtbmV0ICwgcGFyYSBlbmNvbnRyYSBkZSBtYW5lcmEgYXV0b23DoXRpY2F1bmEgZWwgdmFsb3Igw7NwdGltbyBkZSBsYW1iZGEuDQoNCmBgYHtyfQ0KbGlicmFyeShnbG1uZXQpDQpsaWJyYXJ5KE1hdHJpeCkNCnNldC5zZWVkKDEyMykgDQoNCnhfdHJhaW4gPC0gbW9kZWwubWF0cml4KHJlaG9zcF9vbXN+LiwgdHJhaW5pbmcpWywtMV0NCnlfdHJhaW4gPC0gaWZlbHNlKHRyYWluaW5nJHJlaG9zcF9vbXMgPT0gIjEiLCAxLCAwKQ0KDQojI0hhbGxhbmRvIGVsIG1lam9yIGxhbWJkYQ0KY3YubGFzc28gPC0gY3YuZ2xtbmV0KHhfdHJhaW4sIHlfdHJhaW4sIGZhbWlseSA9ICJiaW5vbWlhbCIsIHR5cGUubWVhc3VyZSA9ICJtc2UiLCBhbHBoYSA9IDEpDQpwbG90KGN2Lmxhc3NvKQ0KYGBgDQoNCkxhIGdyw6FmaWNhIG11ZXN0cmEgcXVlIGVsIHJlZ2lzdHJvIGRlbCB2YWxvciDDs3B0aW1vIGRlIGxhbWJkYSAoZXMgZGVjaXIsIGVsIHF1ZSBtaW5pbWl6YSBlbCBlcnJvciBjdWFkcsOhdGljbyBtZWRpbykgZXMgYXByb3hpbWFkYW1lbnRlIC0zLiBFbCB2YWxvciBleGFjdG8gc2UgcHVlZGUgdmVyIGFsIGV4YW1pbmFyIGxhIHZhcmlhYmxlIGxhbWJkYV9taW4gZW4gZWwgY8OzZGlnbyBhIGNvbnRpbnVhY2nDs24uIEVuIGdlbmVyYWwsIHNpbiBlbWJhcmdvLCBlbCBvYmpldGl2byBkZSBsYSByZWd1bGFyaXphY2nDs24gZXMgZXF1aWxpYnJhciBsYSBwcmVjaXNpw7NuIHkgbGEgc2ltcGxpY2lkYWQuIEVuIGVsIGNvbnRleHRvIGFjdHVhbCwgZXN0byBzaWduaWZpY2EgdW4gbW9kZWxvIGNvbiBlbCBtZW5vciBuw7ptZXJvIGRlIGNvZWZpY2llbnRlcyBxdWUgdGFtYmnDqW4gcHJvcG9yY2lvbmEgdW5hIGJ1ZW5hIHByZWNpc2nDs24gLiANCg0KYGBge3J9DQpjdi5sYXNzbyRsYW1iZGEubWluDQpgYGANCg0KRW4gZ2VuZXJhbCwgZWwgcHJvcMOzc2l0byBkZSBsYSByZWd1bGFyaXphY2nDs24gZXMgZXF1aWxpYnJhciBsYSBwcmVjaXNpw7NuIHkgbGEgc2ltcGxpY2lkYWQuIEVzdG8gc2lnbmlmaWNhLCB1biBtb2RlbG8gY29uIGVsIG1lbm9yIG7Dum1lcm8gZGUgcHJlZGljdG9yZXMgcXVlIHRhbWJpw6luIGRhIHVuYSBidWVuYSBwcmVjaXNpw7NuLiBQYXJhIGVzdGUgZmluLCBsYSBmdW5jacOzbiBjdi5nbG1uZXQoKSB0YW1iacOpbiBlbmN1ZW50cmEgZWwgdmFsb3IgZGUgbGFtYmRhIHF1ZSBwcm9wb3JjaW9uYSBlbCBtb2RlbG8gbcOhcyBzaW1wbGUgcGVybyB0YW1iacOpbiBzZSBlbmN1ZW50cmEgZGVudHJvIGRlIHVuIGVycm9yIGVzdMOhbmRhciBkZWwgdmFsb3Igw7NwdGltbyBkZSBsYW1iZGEuIEVzdGUgdmFsb3Igc2UgbGxhbWEgbGFtYmRhLjFzZS4gRXN0ZSB2YWxvciBkZSBsYW1iZGEgKCBsYW1iZGEuMXNlKSBlcyBsbyBxdWUgdXNhcmVtb3MgZW4gZWwgcmVzdG8gZGUgbGEgY29tcHV0YWNpw7NuLg0KDQoNCmBgYHtyfQ0KY3YubGFzc28kbGFtYmRhLjFzZQ0KYGBgDQoNCmBgYHtyfQ0KY29lZihjdi5sYXNzbywgY3YubGFzc28kbGFtYmRhLm1pbikNCmBgYA0KDQpMYSBzYWxpZGEgbXVlc3RyYSBxdWUgc29sbyBhcXVlbGxhcyB2YXJpYWJsZXMgcXVlIGhlbW9zIGRldGVybWluYWRvIHF1ZSBzb24gc2lnbmlmaWNhdGl2YXMgZW4gYmFzZSBhIGxvcyB2YWxvcmVzIGRlIHAgdGllbmVuIGNvZWZpY2llbnRlcyBkaXN0aW50b3MgZGUgY2VybywgZW4gZXN0ZSBjYXNvIGVsIHBhZ28gaG9zcGl0YWxhcmlvLCB0b2RvcyBsb3MgY29lZmljaWVudGVzIGRlIGxhcyBkZW3DoXMgdmFyaWFibGVzIGhhbiBzaWRvIHB1ZXN0b3MgYSBjZXJvIHBvciBlbCBhbGdvcml0bW8uDQoNClVzYW5kbyBsYW1iZGEuMXNlY29tbyBsYSBtZWpvciBsYW1iZGEsIGRhIGxvcyBzaWd1aWVudGVzIGNvZWZpY2llbnRlcyBkZSByZWdyZXNpw7NuOg0KDQpgYGB7cn0NCmNvZWYoY3YubGFzc28sIGN2Lmxhc3NvJGxhbWJkYS4xc2UpDQpgYGANCg0KVXNhbmRvIGxhbWJkYS4xc2UsIGxvcyBjb2VmaWNpZW50ZSBkZSA0IHZhcmlhYmxlcyBzZSBoYW4gZXN0YWJsZWNpZG8gZW4gY2VybyBtZWRpYW50ZSBlbCBhbGdvcml0bW8gZGUgbGF6bywgcmVkdWNpZW5kbyBsYSBjb21wbGVqaWRhZCBkZWwgbW9kZWxvLg0KDQpMYSBjb25maWd1cmFjacOzbiBkZSBsYW1iZGEgPSBsYW1iZGEuMXNlIHByb2R1Y2UgdW4gbW9kZWxvIG3DoXMgc2ltcGxlIGVuIGNvbXBhcmFjacOzbiBjb24gbGFtYmRhLm1pbiwgcGVybyBlbCBtb2RlbG8gcG9kcsOtYSBzZXIgdW4gcG9jbyBtZW5vcyBwcmVjaXNvIHF1ZSBlbCBvYnRlbmlkbyBjb24gbGFtYmRhLm1pbi4NCg0KYGBge3J9DQojI0xhbWJkYSByZWdyZXNzaW9uDQpzdGRfcmlkZ2VfbG9naXQgPC0gZ2xtbmV0KHhfdHJhaW4sIHlfdHJhaW4sIGZhbWlseT0iYmlub21pYWwiLCBhbHBoYT0xKQ0KU1JMX3ByZWRfdHJhaW4gPC0gcHJlZGljdChzdGRfcmlkZ2VfbG9naXQsIHhfdHJhaW4sIHR5cGU9ImNsYXNzIiwgcz1jdi5sYXNzbyRsYW1iZGEuMXNlKQ0KDQpgYGANCg0KIyMjTWF0cml6IHRyYWluaW5nDQpgYGB7cn0NCmNvbmZ1c2lvbl9tYXRyaXhfdHJhaW4gPC0gdGFibGUoeV90cmFpbiwgU1JMX3ByZWRfdHJhaW4pDQpjb25mdXNpb25fbWF0cml4X3RyYWluDQpgYGANCg0KYGBge3J9DQplcnJvcl9yYXRlX3RyYWluIDwtICg3NjArOTIwKS8oNzYwKzkyMCsxODE0KzEzNjgpDQplcnJvcl9yYXRlX3RyYWluDQpgYGANCg0KIyMjTWF0cml6IHRlc3QNCmBgYHtyfQ0KDQp4X3Rlc3QgPC0gbW9kZWwubWF0cml4KHJlaG9zcF9vbXN+LiwgdGVzdGluZylbLC0xXQ0KeV90ZXN0IDwtIGlmZWxzZSh0ZXN0aW5nJHJlaG9zcF9vbXMgPT0gMSwgMSwgMCkNCg0KDQpgYGANCg0KYGBge3J9DQpTUkxfcHJlZF90ZXN0IDwtIHByZWRpY3Qoc3RkX3JpZGdlX2xvZ2l0LCB4X3Rlc3QsIHR5cGU9ImNsYXNzIiwgcz1jdi5sYXNzbyRsYW1iZGEuMXNlKQ0KY29uZnVzaW9uX21hdHJpeF90ZXN0IDwtIHRhYmxlKHlfdGVzdCwgU1JMX3ByZWRfdGVzdCkNCmNvbmZ1c2lvbl9tYXRyaXhfdGVzdA0KYGBgDQoNCmBgYHtyfQ0KZXJyb3JfcmF0ZV90ZXN0IDwtICg4NCsyODY3KS8oNzM0NCsyODY3Kzg0KzEyOSkNCmVycm9yX3JhdGVfdGVzdA0KYGBgDQoNCkNhbGN1bGFuZG8gZWwgbW9kZWxvIGNvbiByaWRnZToNCmBgYHtyfQ0KIyNIYWxsYW5kbyBlbCBtZWpvciBsYW1iZGENCmN2LnJpZGdlIDwtIGN2LmdsbW5ldCh4X3RyYWluLCB5X3RyYWluLCBmYW1pbHkgPSAiYmlub21pYWwiLCB0eXBlLm1lYXN1cmUgPSAibXNlIiwgYWxwaGEgPSAwKQ0KY29lZihjdi5yaWRnZSwgY3YucmlkZ2UkbGFtYmRhLjFzZSkNCmBgYA0KDQpgYGB7cn0NClNSUl9wcmVkX3Rlc3QgPC0gcHJlZGljdChzdGRfcmlkZ2VfbG9naXQsIHhfdGVzdCwgdHlwZT0iY2xhc3MiLCBzPWN2LnJpZGdlJGxhbWJkYS4xc2UpDQpjb25mdXNpb25fbWF0cml4X3Rlc3QgPC0gdGFibGUoeV90ZXN0LCBTUlJfcHJlZF90ZXN0KQ0KY29uZnVzaW9uX21hdHJpeF90ZXN0DQpgYGANCg0KDQpbVm9sdmVyIGFsIMONbmRpY2VdKCNpbmRpY2UpPC9kaXY+DQoNCjxocj4NCjxoMyBpZD0iUmVjb21lbmRhY2lvbiIgc3R5bGU9InRleHQtYWxpZ246IGNlbnRlcjsiIG1hcmtkb3duPSIxIj5SZWNvbWVuZGFjaW9uZXMgeSBFc3RyYXRlZ2lhczwvaDM+DQo8ZGl2IHN0eWxlPSJ0ZXh0LWFsaWduOiBqdXN0aWZ5Ij4NCg0KQ29tbyBzZSBwdWRvIG9ic2VydmFyIGVuIGxhIGRlZmluaWNpb24gZGUgbGEgdmFyaWFibGUgZW5kw7NnZW5hIGRlIGxhIHJlaG9zcGl0YWxpemFjacOzbiwgZXN0YSBzZSBjb25zdHJ1ecOzIG1lZGlhbnRlIGRvcyByZXN0cmljY2lvbmVzIGVuIGVsIHNldCBkZSBkYXRvcyBpbmljaWFsOg0KMS4gUXVlIGRlbnRybyBkZSBsb3Mgc2lndWllbnRlcyAzMCBkaWFzIGEgbGEgcHJpbWVyYSBob3NwaXRhbGl6YWNpw7NuIHN1cmdpZXJhIHVuYSBzZWd1bmRhIGhvc3BpdGFsaXphY2lvbi4NCjIuIFF1ZSBwYXJhIGFxdWVsbG9zIGNhc29zIGRvbmRlIGhheSBkb3MgZXZlbnRvcyBlbiBsYSB2ZW50YW5hIGRlIDMwIGTDrWFzLCBsb3MgY8OzZGlnb3MgZGUgZGlhZ27Ds3N0aWNvIENJRTEwIGRlbCBwcmltZXJvIHkgc2VndW5kbyBldmVudG8gcGVydGVuZWNpZXJhbiBhIGxhIG1pc21hIGNhdGVnb3LDrWEgZGUgZGlhZ8Ozc3RpY28gZW4gbGEgY2xhc2lmaWNhY2nDs24gZGUgbGEgT01TLg0KDQpFbiBlc3RlIHNlbnRpZG8sIHRlbmVtb3MgZG9zIGNvbmRpY2lvbmFudGVzIHF1ZSBzZSBwb2Ryw61hbiBhanVzdGFyLCB5YSBxdWUgcmVzdHJpbmdlbiBsYSBwb3NpYmlsaWRhZCBkZSBlbmNvbnRyYXIgdW4gbWF5b3IgbnVtZXJvIGRlIGNhc29zIHF1ZSBzZSBwdWVkYW4gY29uc2lkZXJhciBjb21vIHJlaG9zcGl0YWxpemFjacOzbjsgcG9yIGVqZW1wbG8sIGFsZ3Vub3MgZXN0dWRpb3Mgc3VnaWVyZW4gcXVlIGxhIHZlbnRhbmEgZGUgdGllbXBvIHBvZHLDrWEgc2VyIGRlIDE1IGRpYXMgZW50cmUgZWwgcHJpbWVyIHkgc2VndW5kbyBldmVudG8uIEVuIGN1YW50byBhIGxhIHNpbWlsaXR1ZCBkZSBsb3MgZGlhZ27Ds3N0aWNvcyBkZSBhbWJvcyBldmVudG9zLCBwYXJhIGVzdGUgbW9kZWxvIHNlIHR1dm8gZW4gY3VlbnRhIHNvbG8gc2kgYW1ib3MgZGlhZ27Ds3N0aWNvcyBwZXJ0ZW5lY2VuIGEgbGEgbWlzbWEgY2F0ZWdvcsOtYSwgc2luIGVtYmFyZ28gZXMgaW1wb3J0YW50ZSB0ZW5lciBlbiBjdWVudGEgcXVlIGhheSBkaWFnbsOzc3RpY29zIHF1ZSBwdWVkZW4gZGVzZW5jYWRlbmFyIGVuIG90cm9zIHF1ZSBubyBuZWNlc2FyaWFtZW50ZSBzZWFuIGRlIGxhIG1pc21hIGNhdGVnb3LDrWEuIEVzdGUgdGlwbyBkZSBhc29jaWFjaW9uZXMgcmVxdWllcmVuIGRlIHVuIG1heW9yIGFuw6FsaXNpcyBhIG5pdmVsIG3DqWRpY28uDQoNClNlIGhhbiBkZXNjcml0byBmYWN0b3JlcyBhc29jaWFkb3MgY29uIGxhIHJlaG9zcGl0YWxpemFjaW9uIHJlbGFjaW9uYWRvcyBjb24gbGEgY2FsaWRhZCBkZSB2aWRhIGRlIGxvcyBwYWNpZW50ZXMsIHNpbiBlbWJhcmdvIGVuIGVzdGUgbW9kZWxvIG5vIHNlIGluY2x1eW8gZ3JhbiBwYXJ0ZSBkZSBlc3RlIHRpcG8gZGUgaW5mb3JtYWNpb24sIHlhIHF1ZSwgZGVzZGUgZWwgcHJpbmNpcGlvLCBsYSBwb2JsYWNpw7NuIG9iamV0aXZvIHNlIGNvbXBvbmUgZGUgaW5kaXZpZHVvcyBhc2VndXJhZG9zIGVuIHBvbGl6YSBkZSB2aWRhIHkgc2FsdWQsIGVuIHN1IG1heW9yaWEsIGRlIGVzdHJhdG8gY3VhdHJvIGhhY2lhIGFycmliYS4gRGUgZXN0YSBmb3JtYSB5YSBubyBhcG9ydGEgaW5mb3JtYWNpw7NuIG1lZGlyIGVsIG5pdmVsIGRlIGNhbGlkYWQgZGUgdmlkYSBvIGZhbGljaWRhZCBkZSBhY2Nlc28gYSBsb3Mgc2VydmljaW9zIGRlIHNhbHVkLCBwZXJvIHNlIHBvZHJpYSB0ZW5lciBlbiBjdWVudGEgaW5mb3JtYWNpw7NuIHJlbGFjaW9uYWRhIGNvbiBzaW50b21hcyBkZXByZXNpdm9zLg0KDQpMYSBjYWxpZGFkIGVuIGVsIGN1aWRhZG8gaG9zcGl0YWxhcmlvIHRhbWJpZW4gc2UgaGEgY29uc2lkZXJhZG8gY29tbyB1biBmYWN0b3IgaW1wb3J0YW50ZSwgcG9yIGxvIHRhbnRvIHNlIHBvZHLDrWEgY29uc2lkZXJhciBsYSBhZ3JlZ2FjacOzbiBkZSAgaW5mb3JtYWNpw7NuIHF1ZSBpbmRpcXVlIHF1ZSB0YW4gw7NwdGltYXMgc29uIGxhcyBjb25kaWNpb25lcyBwYXJhIHVuYSBidWVuYSBhdGVuY2nDs24gZW4gbG9zIGNlbnRyb3MgaG9zcGl0YWxhcmlvcyB5IGx1Z2FyZXMgcXVlIHNlIHR1dmllcm9uIGVuIGN1ZW50YS4NCg0KYGBge3J9DQojIFBBU08gMTogICBDYXJnYSBQYWNrYWdlIHkgU2V0IGRlIGRhdG9zDQojIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KbGlicmFyeShDNTApDQpsaWJyYXJ5KHJwYXJ0KQ0KbGlicmFyeShycGFydC5wbG90KSANCmRhdGEoY2h1cm4pOyAjIGNhcmdhIHRhYmxhcw0KDQojIFBBU08gMjogICBDcmVhIEFyYm9sIGRlIERlY2lzaW9uDQojIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KTW9kZWxvQXJib2wgPC0gcnBhcnQocmVob3NwX29tcyB+IC4sIGRhdGEgPSB0cmFpbmluZywgcGFybXM9bGlzdChzcGxpdD0iaW5mb3JtYXRpb24iKSkNCg0KIyBQQVNPIDM6ICBQcmVkaWNlIHJlaG9zcGl0YWxpemFjaW9uIGVuIGRhdG9zIGRlIFRFU1QNCiMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQpQcmVkaWNjaW9uIDwtIHByZWRpY3QoTW9kZWxvQXJib2wsIHRlc3RpbmcsIHR5cGU9ImNsYXNzIikgIyBQcmVkaWNjacOzbiBlbiBUZXN0DQpNQyAgICAgICAgIDwtIHRhYmxlKHRlc3RpbmckcmVob3NwX29tcywgUHJlZGljY2lvbikgIyBNYXRyaXogZGUgQ29uZnVzacOzbg0KDQpNQw0KIyBQQVNPIDQ6IENyZWEgR3JhZmljbw0KIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCnJwYXJ0LnBsb3QoTW9kZWxvQXJib2wsIHR5cGU9MSwgZXh0cmE9MTAwLGNleCA9IC43LCAgYm94LmNvbD1jKCJncmF5OTkiLCAiZ3JheTg4IilbTW9kZWxvQXJib2wkZnJhbWUkeXZhbF0pDQpgYGANCg0KDQpgYGB7cn0NCmxpYnJhcnkoZTEwNzEpDQoNCiMgRWplY3VjacOzbiBkZWwgbW9kZWxvIFNWTQ0KbW9kZWxvc3ZtIDwtIHN2bShyZWhvc3Bfb21zIH4gLiwgZGF0YSA9IHRyYWluaW5nKQ0KDQojIFByZWRpY2Npw7NuIGRlIGxvcyByZXN0YW50ZXMNCnByZWRpY2Npb25zdm0gPC0gcHJlZGljdChtb2RlbG9zdm0sIG5ldyA9IHRlc3RpbmcpDQoNCiMgVGFibGEgZGUgY29uZnVzacOzbi4NCiMgU2UgdXNhIHdpdGggcGFyYSBxdWUgYXBhcmV6Y2EgZWwgbm9tYnJlIGRlIGxhIHZhcmlhYmxlIFNwZWNpZXMgZW4gZWxsYQ0KIyB5YSBxdWUgZW4gY2FzbyBjb250cmFyaW8gbm8gc2FsZS4NCihtYyA8LSB3aXRoKHRlc3RpbmcsKHRhYmxlKHByZWRpY2Npb25zdm0sIHJlaG9zcF9vbXMpKSkpDQoNCg0KIyAlIGNvcnJlY3RhbWVudGUgY2xhc2lmaWNhZG9zDQooY29ycmVjdG9zIDwtIHN1bShkaWFnKG1jKSkgLyBucm93KHRlc3RpbmcpICoxMDApDQoNCg0KYGBgDQoNCg0KYGBge3J9DQpsaWJyYXJ5KGlwcmVkKQ0KDQojIEVqZWN1Y2nDs24gZGVsIG1vZGVsbyBkZSBCYWdnaW5nDQptb2RlbG9iYWcgPC0gYmFnZ2luZyhyZWhvc3Bfb21zfi4sIGRhdGE9dHJhaW5pbmcpDQoNCiMgUmVzdW1lbiBkZWwgYWp1c3RlIGRlbCBtb2RlbG8NCiNtb2RlbG8NCiMjIA0KIyMgQmFnZ2luZyBjbGFzc2lmaWNhdGlvbiB0cmVlcyB3aXRoIDI1IGJvb3RzdHJhcCByZXBsaWNhdGlvbnMgDQojIyANCiMjIENhbGw6IGJhZ2dpbmcuZGF0YS5mcmFtZShmb3JtdWxhID0gU3BlY2llcyB+IC4sIGRhdGEgPSBkYXRvcy5lbnRyZW5vKQ0KIyBIYWNlciBwcmVkaWNjaW9uZXMNCnByZWRpY2Npb25iYWcgPC0gcHJlZGljdChtb2RlbG9iYWcsIHRlc3RpbmcpDQoNCiMgTWF0cml6IGRlIGNvbmZ1c2nDs24NCihtYyA8LSB3aXRoKHRlc3RpbmcsdGFibGUocHJlZGljY2lvbmJhZywgcmVob3NwX29tcykpKQ0KDQoNCmBgYA0KDQoNCmBgYHtyfQ0KIyBDYXJnYSBlbCBwYXF1ZXRlIGVzcGVjw61maWNvIGRlbCBtw6l0b2RvIFJhbmRvbSBGb3Jlc3QNCmxpYnJhcnkocmFuZG9tRm9yZXN0KQ0KDQojIEFqdXN0YXIgbW9kZWxvDQptb2RlbG9SRm9yZXN0IDwtIHJhbmRvbUZvcmVzdChyZWhvc3Bfb21zfi4sIGRhdGE9dHJhaW5pbmcpDQoNCiMgUmVzdW1lbiBkZWwgYWp1c3RlIGRlbCBtb2RlbG8NCiNtb2RlbG8NCg0KcHJlZGljY2lvblJGb3Jlc3QgPC0gcHJlZGljdChtb2RlbG9SRm9yZXN0LCB0ZXN0aW5nKQ0KDQojIE1hdHJpeiBkZSBjb25mdXNpw7NuDQoobWMgPC0gd2l0aCh0ZXN0aW5nLCB0YWJsZShwcmVkaWNjaW9uUkZvcmVzdCwgcmVob3NwX29tcykpKQ0KDQpgYGA=